ItemsControlTests.cs 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Collections.ObjectModel;
  5. using System.Collections.Specialized;
  6. using System.Linq;
  7. using Avalonia.Collections;
  8. using Avalonia.Controls.Presenters;
  9. using Avalonia.Controls.Primitives;
  10. using Avalonia.Controls.Templates;
  11. using Avalonia.Data;
  12. using Avalonia.Headless;
  13. using Avalonia.Input;
  14. using Avalonia.Layout;
  15. using Avalonia.LogicalTree;
  16. using Avalonia.Markup.Xaml.Templates;
  17. using Avalonia.Media;
  18. using Avalonia.Styling;
  19. using Avalonia.UnitTests;
  20. using Avalonia.VisualTree;
  21. using Xunit;
  22. #nullable enable
  23. namespace Avalonia.Controls.UnitTests
  24. {
  25. public class ItemsControlTests : ScopedTestBase
  26. {
  27. [Fact]
  28. public void Setting_ItemsSource_Should_Populate_Items()
  29. {
  30. using var app = Start();
  31. var target = CreateTarget(itemsSource: new[] { "foo", "bar" });
  32. Assert.NotSame(target.ItemsSource, target.Items);
  33. Assert.Equal(target.ItemsSource, target.Items);
  34. }
  35. [Fact]
  36. public void Cannot_Set_ItemsSource_With_Items_Present()
  37. {
  38. using var app = Start();
  39. var target = CreateTarget();
  40. target.Items.Add("foo");
  41. Assert.Throws<InvalidOperationException>(() => target.ItemsSource = new[] { "baz" });
  42. }
  43. [Fact]
  44. public void Cannot_Modify_Items_When_ItemsSource_Set()
  45. {
  46. using var app = Start();
  47. var target = CreateTarget(itemsSource: Array.Empty<string>());
  48. Assert.Throws<InvalidOperationException>(() => target.Items.Add("foo"));
  49. }
  50. [Fact]
  51. public void Should_Use_ItemTemplate_To_Create_Control()
  52. {
  53. using var app = Start();
  54. var target = CreateTarget(
  55. itemsSource: new[] { "Foo" },
  56. itemTemplate: new FuncDataTemplate<string>((_, __) => new Canvas()));
  57. var container = GetContainer(target);
  58. Assert.IsType<Canvas>(container.Child);
  59. }
  60. [Fact]
  61. public void ItemTemplate_Can_Be_Changed()
  62. {
  63. using var app = Start();
  64. var target = CreateTarget(
  65. itemsSource: new[] { "Foo" },
  66. itemTemplate: new FuncDataTemplate<string>((_, __) => new Canvas()));
  67. var container = GetContainer(target);
  68. Assert.IsType<Canvas>(container.Child);
  69. target.ItemTemplate = new FuncDataTemplate<string>((_, __) => new Border());
  70. Layout(target);
  71. container = GetContainer(target);
  72. Assert.IsType<Border>(container.Child);
  73. }
  74. [Fact]
  75. public void Panel_Should_Have_TemplatedParent_Set_To_ItemsControl()
  76. {
  77. using var app = Start();
  78. var target = CreateTarget(itemsSource: new[] { "Foo" });
  79. Assert.Equal(target, target.ItemsPanelRoot?.TemplatedParent);
  80. }
  81. [Fact]
  82. public void Panel_Should_Have_ItemsHost_Set_To_True()
  83. {
  84. using var app = Start();
  85. var target = CreateTarget(itemsSource: new[] { "Foo" });
  86. Assert.True(target.ItemsPanelRoot?.IsItemsHost);
  87. }
  88. [Fact]
  89. public void Container_Should_Have_TemplatedParent_Set_To_Null()
  90. {
  91. using var app = Start();
  92. var target = CreateTarget(itemsSource: new[] { "Foo" });
  93. var container = GetContainer(target);
  94. Assert.Null(container.TemplatedParent);
  95. }
  96. [Fact]
  97. public void Container_Should_Have_Theme_Set_To_ItemContainerTheme()
  98. {
  99. using var app = Start();
  100. var theme = new ControlTheme { TargetType = typeof(ContentPresenter) };
  101. var target = CreateTarget(
  102. itemsSource: new[] { "Foo" },
  103. itemContainerTheme: theme);
  104. var container = GetContainer(target);
  105. Assert.Same(container.Theme, theme);
  106. }
  107. [Fact]
  108. public void Container_Should_Have_Theme_Set_To_ItemContainerTheme_With_Base_TargetType()
  109. {
  110. using var app = Start();
  111. var theme = new ControlTheme { TargetType = typeof(Control) };
  112. var target = CreateTarget(
  113. itemsSource: new[] { "Foo" },
  114. itemContainerTheme: theme);
  115. var container = GetContainer(target);
  116. Assert.Same(container.Theme, theme);
  117. }
  118. [Fact]
  119. public void ItemContainerTheme_Can_Be_Changed()
  120. {
  121. using var app = Start();
  122. var theme1 = new ControlTheme
  123. {
  124. TargetType = typeof(ContentPresenter),
  125. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Red) }
  126. };
  127. var theme2 = new ControlTheme
  128. {
  129. TargetType = typeof(ContentPresenter),
  130. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Green) }
  131. };
  132. var target = CreateTarget(
  133. itemsSource: new[] { "Foo" },
  134. itemContainerTheme: theme1);
  135. var container = GetContainer(target);
  136. Assert.Same(container.Theme, theme1);
  137. Assert.Equal(container.Background, Brushes.Red);
  138. target.ItemContainerTheme = theme2;
  139. container = GetContainer(target);
  140. Assert.Same(container.Theme, theme2);
  141. Assert.Equal(container.Background, Brushes.Green);
  142. }
  143. [Fact]
  144. public void ItemContainerTheme_Can_Be_Changed_Virtualizing()
  145. {
  146. using var app = Start();
  147. var theme1 = new ControlTheme
  148. {
  149. TargetType = typeof(ContentPresenter),
  150. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Red) }
  151. };
  152. var theme2 = new ControlTheme
  153. {
  154. TargetType = typeof(ContentPresenter),
  155. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Green) }
  156. };
  157. var itemsPanel = new FuncTemplate<Panel?>(() => new VirtualizingStackPanel());
  158. var target = CreateTarget(
  159. itemsSource: new[] { "Foo" },
  160. itemContainerTheme: theme1,
  161. itemsPanel: itemsPanel);
  162. var container = GetContainer(target);
  163. Assert.Same(container.Theme, theme1);
  164. Assert.Equal(container.Background, Brushes.Red);
  165. target.ItemContainerTheme = theme2;
  166. Layout(target);
  167. container = GetContainer(target);
  168. Assert.Same(container.Theme, theme2);
  169. Assert.Equal(container.Background, Brushes.Green);
  170. }
  171. [Fact]
  172. public void ItemContainerTheme_Can_Be_Cleared()
  173. {
  174. using var app = Start();
  175. var theme = new ControlTheme
  176. {
  177. TargetType = typeof(ContentPresenter),
  178. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Red) }
  179. };
  180. var target = CreateTarget(
  181. itemsSource: new[] { "Foo" },
  182. itemContainerTheme: theme);
  183. var container = GetContainer(target);
  184. Assert.Same(container.Theme, theme);
  185. Assert.Equal(container.Background, Brushes.Red);
  186. target.ItemContainerTheme = null;
  187. container = GetContainer(target);
  188. Assert.Null(container.Theme);
  189. Assert.Null(container.Background);
  190. }
  191. [Fact]
  192. public void ItemContainerTheme_Should_Not_Override_LocalValue_Theme()
  193. {
  194. using var app = Start();
  195. var theme1 = new ControlTheme
  196. {
  197. TargetType = typeof(ContentPresenter),
  198. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Red) }
  199. };
  200. var theme2 = new ControlTheme
  201. {
  202. TargetType = typeof(Control),
  203. Setters = { new Setter(ContentPresenter.BackgroundProperty, Brushes.Green) }
  204. };
  205. var items = new object[]
  206. {
  207. new ContentPresenter(),
  208. new ContentPresenter
  209. {
  210. Theme = theme2
  211. },
  212. };
  213. var target = CreateTarget(
  214. itemsSource: items,
  215. itemContainerTheme: theme1);
  216. Assert.Same(theme1, GetContainer(target, 0).Theme);
  217. Assert.Same(theme2, GetContainer(target, 1).Theme);
  218. target.ItemContainerTheme = null;
  219. Assert.Null(GetContainer(target, 0).Theme);
  220. Assert.Same(theme2, GetContainer(target, 1).Theme);
  221. }
  222. [Fact]
  223. public void Container_Should_Have_LogicalParent_Set_To_ItemsControl()
  224. {
  225. using var app = UnitTestApplication.Start(TestServices.StyledWindow);
  226. var target = new ItemsControl();
  227. var root = CreateRoot(target);
  228. var templatedParent = new Button();
  229. target.TemplatedParent = templatedParent;
  230. target.Template = CreateItemsControlTemplate();
  231. target.ItemsSource = new[] { "Foo" };
  232. root.LayoutManager.ExecuteInitialLayoutPass();
  233. var container = GetContainer(target);
  234. Assert.Equal(target, container.Parent);
  235. }
  236. [Fact]
  237. public void Control_Item_Should_Be_Logical_Child_Before_ApplyTemplate()
  238. {
  239. using var app = Start();
  240. var child = new Control();
  241. var target = CreateTarget(items: new[] { child }, performLayout: false);
  242. Assert.False(target.IsMeasureValid);
  243. Assert.Empty(target.GetVisualChildren());
  244. Assert.Equal(child.Parent, target);
  245. Assert.Equal(child.GetLogicalParent(), target);
  246. Assert.Equal(new[] { child }, target.GetLogicalChildren());
  247. }
  248. [Fact]
  249. public void Control_Item_Should_Be_Logical_Child_After_Layout()
  250. {
  251. using var app = Start();
  252. var child = new Control();
  253. var target = CreateTarget(items: new[] { child });
  254. Assert.True(target.IsMeasureValid);
  255. Assert.Single(target.GetVisualChildren());
  256. Assert.Equal(target, child.Parent);
  257. Assert.Equal(target, child.GetLogicalParent());
  258. Assert.Equal(new[] { child }, target.GetLogicalChildren());
  259. }
  260. [Fact]
  261. public void Added_Container_Should_Have_LogicalParent_Set_To_ItemsControl()
  262. {
  263. using var app = Start();
  264. var items = new ObservableCollection<Border>();
  265. var target = CreateTarget(itemsSource: items);
  266. var item = new Border();
  267. items.Add(item);
  268. Assert.Equal(target, item.Parent);
  269. }
  270. [Fact]
  271. public void Control_Item_Can_Be_Removed_From_Logical_Children_Before_ApplyTemplate()
  272. {
  273. using var app = Start();
  274. var child = new Control();
  275. var target = CreateTarget(items: new[] { child }, performLayout: false);
  276. Assert.False(target.IsMeasureValid);
  277. Assert.Empty(target.GetVisualChildren());
  278. Assert.Single(target.GetLogicalChildren());
  279. target.Items.RemoveAt(0);
  280. Assert.Null(child.Parent);
  281. Assert.Null(child.GetLogicalParent());
  282. Assert.Empty(target.GetLogicalChildren());
  283. }
  284. [Fact]
  285. public void Clearing_Items_Should_Clear_Child_Controls_Parent_Before_ApplyTemplate()
  286. {
  287. using var app = Start();
  288. var child = new Control();
  289. var target = CreateTarget(items: new[] { child }, performLayout: false);
  290. Assert.False(target.IsMeasureValid);
  291. Assert.Empty(target.GetVisualChildren());
  292. Assert.Single(target.GetLogicalChildren());
  293. target.Items.Clear();
  294. Assert.Null(child.Parent);
  295. Assert.Null(child.GetLogicalParent());
  296. }
  297. [Fact]
  298. public void Assigning_ItemsSource_Should_Not_Fire_LogicalChildren_CollectionChanged_Before_ApplyTemplate()
  299. {
  300. using var app = Start();
  301. var child = new Control();
  302. var target = CreateTarget(itemsSource: new[] { child }, performLayout: false);
  303. var called = false;
  304. ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) => called = true;
  305. var list = new AvaloniaList<Control>(new[] { child });
  306. target.ItemsSource = list;
  307. Assert.False(called);
  308. }
  309. [Fact]
  310. public void Removing_ItemsSource_Items_Should_Not_Fire_LogicalChildren_CollectionChanged_Before_ApplyTemplate()
  311. {
  312. using var app = Start();
  313. var items = new AvaloniaList<string> { "Foo", "Bar" };
  314. var target = CreateTarget(itemsSource: items, performLayout: false);
  315. var called = false;
  316. ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) => called = true;
  317. items.Remove("Bar");
  318. Assert.False(called);
  319. }
  320. [Fact]
  321. public void Changing_ItemsSource_Should_Not_Fire_LogicalChildren_CollectionChanged_Before_ApplyTemplate()
  322. {
  323. using var app = Start();
  324. var child = new Control();
  325. var target = CreateTarget(itemsSource: new[] { child }, performLayout: false);
  326. var called = false;
  327. ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) => called = true;
  328. var list = new AvaloniaList<Control>();
  329. target.ItemsSource = list;
  330. list.Add(child);
  331. Assert.False(called);
  332. }
  333. [Fact]
  334. public void Clearing_Items_Should_Clear_Child_Controls_Parent()
  335. {
  336. using var app = Start();
  337. var child = new Control();
  338. var target = CreateTarget(items: new[] { child });
  339. target.Items.Clear();
  340. Assert.Null(child.Parent);
  341. Assert.Null(((ILogical)child).LogicalParent);
  342. }
  343. [Fact]
  344. public void Adding_Control_Item_Should_Make_Control_Appear_In_LogicalChildren()
  345. {
  346. using var app = Start();
  347. var child = new Control();
  348. var target = CreateTarget(items: new[] { child }, performLayout: false);
  349. // Should appear both before and after applying template.
  350. Assert.Equal(new ILogical[] { child }, target.GetLogicalChildren());
  351. Layout(target);
  352. Assert.Equal(new ILogical[] { child }, target.GetLogicalChildren());
  353. }
  354. [Fact]
  355. public void Adding_String_Item_Should_Make_ContentPresenter_Appear_In_LogicalChildren()
  356. {
  357. using var app = Start();
  358. var target = CreateTarget(itemsSource: new[] { "Foo " });
  359. var logical = (ILogical)target;
  360. Assert.Equal(1, logical.LogicalChildren.Count);
  361. Assert.IsType<ContentPresenter>(logical.LogicalChildren[0]);
  362. }
  363. [Fact]
  364. public void Adding_Items_Should_Fire_LogicalChildren_CollectionChanged()
  365. {
  366. using var app = Start();
  367. var target = CreateTarget();
  368. var called = false;
  369. target.Template = CreateItemsControlTemplate();
  370. target.ApplyTemplate();
  371. ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) =>
  372. called = e.Action == NotifyCollectionChangedAction.Add;
  373. var child = new Control();
  374. target.Items.Add(child);
  375. Assert.True(called);
  376. }
  377. [Fact]
  378. public void Clearing_Items_Should_Fire_LogicalChildren_CollectionChanged()
  379. {
  380. using var app = Start();
  381. var child = new Control();
  382. var target = CreateTarget(items: new[] { child });
  383. var called = false;
  384. ((ILogical)target).LogicalChildren.CollectionChanged += (s, e) =>
  385. called = e.Action == NotifyCollectionChangedAction.Remove;
  386. target.Items.Clear();
  387. Assert.True(called);
  388. }
  389. [Fact]
  390. public void LogicalChildren_Should_Not_Change_Instance_When_Template_Changed()
  391. {
  392. using var app = Start();
  393. var target = CreateTarget();
  394. var before = ((ILogical)target).LogicalChildren;
  395. target.Template = null;
  396. target.Template = CreateItemsControlTemplate();
  397. Layout(target);
  398. var after = ((ILogical)target).LogicalChildren;
  399. Assert.NotNull(before);
  400. Assert.NotNull(after);
  401. Assert.Same(before, after);
  402. }
  403. [Fact]
  404. public void Control_Item_Should_Be_Removed_From_LogicalChildren()
  405. {
  406. using var app = Start();
  407. var item = new Border();
  408. var items = new ObservableCollection<Control>();
  409. var target = CreateTarget(itemsSource: items);
  410. items.Add(item);
  411. items.Remove(item);
  412. Assert.Empty(target.LogicalChildren);
  413. }
  414. [Fact]
  415. public void Control_Item_Should_Be_Removed_From_LogicalChildren_Virtualizing()
  416. {
  417. using var app = Start();
  418. var item = new Border();
  419. var items = new ObservableCollection<Control>();
  420. var itemsPanel = new FuncTemplate<Panel?>(() => new VirtualizingStackPanel());
  421. var target = CreateTarget(
  422. itemsPanel: itemsPanel,
  423. itemsSource: items);
  424. items.Add(item);
  425. Layout(target);
  426. items.Remove(item);
  427. Assert.Empty(target.LogicalChildren);
  428. }
  429. [Fact]
  430. public void Should_Clear_Containers_When_ItemsPresenter_Changes()
  431. {
  432. using var app = Start();
  433. var target = CreateTarget(itemsSource: new[] { "foo", "bar" });
  434. var panel = Assert.IsAssignableFrom<Panel>(target.Presenter?.Panel);
  435. Assert.Equal(2, panel.Children.Count());
  436. target.Template = CreateItemsControlTemplate();
  437. target.ApplyTemplate();
  438. Assert.Empty(panel.Children);
  439. }
  440. [Fact]
  441. public void Empty_Class_Should_Initially_Be_Applied()
  442. {
  443. using var app = Start();
  444. var target = CreateTarget(performLayout: false);
  445. Assert.Contains(":empty", target.Classes);
  446. }
  447. [Fact]
  448. public void Empty_Class_Should_Be_Cleared_When_Items_Added()
  449. {
  450. using var app = Start();
  451. var target = CreateTarget(items: new[] { 1, 2, 3 }, performLayout: false);
  452. Assert.DoesNotContain(":empty", target.Classes);
  453. }
  454. [Fact]
  455. public void Empty_Class_Should_Be_Cleared_When_ItemsSource_Items_Added()
  456. {
  457. using var app = Start();
  458. var target = CreateTarget(itemsSource: new[] { 1, 2, 3 }, performLayout: false);
  459. Assert.DoesNotContain(":empty", target.Classes);
  460. }
  461. [Fact]
  462. public void Empty_Class_Should_Be_Set_When_ItemsSource_Collection_Cleared()
  463. {
  464. using var app = Start();
  465. var target = CreateTarget(itemsSource: new[] { 1, 2, 3 });
  466. target.ItemsSource = new int[0];
  467. Assert.Contains(":empty", target.Classes);
  468. }
  469. [Fact]
  470. public void Item_Count_Should_Be_Set_When_ItemsSource_Set()
  471. {
  472. using var app = Start();
  473. var target = CreateTarget(itemsSource: new[] { 1, 2, 3 });
  474. Assert.Equal(3, target.ItemCount);
  475. }
  476. [Fact]
  477. public void Item_Count_Should_Be_Set_When_Items_Changed()
  478. {
  479. using var app = Start();
  480. var items = new ObservableCollection<int>() { 1, 2, 3 };
  481. var target = CreateTarget(items: new[] { 1, 2, 3 });
  482. target.Items.Add(4);
  483. Assert.Equal(4, target.ItemCount);
  484. target.Items.Clear();
  485. Assert.Equal(0, target.ItemCount);
  486. }
  487. [Fact]
  488. public void Item_Count_Should_Be_Set_When_ItemsSource_Items_Changed()
  489. {
  490. using var app = Start();
  491. var items = new ObservableCollection<int>() { 1, 2, 3 };
  492. var target = CreateTarget(itemsSource: items);
  493. items.Add(4);
  494. Assert.Equal(4, target.ItemCount);
  495. items.Clear();
  496. Assert.Equal(0, target.ItemCount);
  497. }
  498. [Fact]
  499. public void Empty_Class_Should_Be_Set_When_Items_Collection_Cleared()
  500. {
  501. using var app = Start();
  502. var items = new ObservableCollection<int>() { 1, 2, 3 };
  503. var target = CreateTarget(itemsSource: items);
  504. items.Clear();
  505. Assert.Contains(":empty", target.Classes);
  506. }
  507. [Fact]
  508. public void Empty_Class_Should_Not_Be_Set_When_ItemsSource_Collection_Count_Increases()
  509. {
  510. using var app = Start();
  511. var items = new ObservableCollection<int>() { };
  512. var target = CreateTarget(itemsSource: items);
  513. items.Add(1);
  514. Assert.DoesNotContain(":empty", target.Classes);
  515. }
  516. [Fact]
  517. public void Single_Item_Class_Should_Be_Set_When_ItemsSource_Collection_Count_Increases_To_One()
  518. {
  519. using var app = Start();
  520. var items = new ObservableCollection<int>() { };
  521. var target = CreateTarget(itemsSource: items);
  522. items.Add(1);
  523. Assert.Contains(":singleitem", target.Classes);
  524. }
  525. [Fact]
  526. public void Empty_Class_Should_Not_Be_Set_When_ItemsSource_Collection_Cleared()
  527. {
  528. using var app = Start();
  529. var items = new ObservableCollection<int>() { 1, 2, 3 };
  530. var target = CreateTarget(itemsSource: items);
  531. items.Clear();
  532. Assert.DoesNotContain(":singleitem", target.Classes);
  533. }
  534. [Fact]
  535. public void Single_Item_Class_Should_Not_Be_Set_When_Items_Collection_Count_Increases_Beyond_One()
  536. {
  537. using var app = Start();
  538. var items = new ObservableCollection<int>() { 1 };
  539. var target = CreateTarget(itemsSource: items);
  540. items.Add(2);
  541. Assert.DoesNotContain(":singleitem", target.Classes);
  542. }
  543. [Fact]
  544. public void DataContexts_Should_Be_Correctly_Set()
  545. {
  546. using var app = Start();
  547. var items = new object[]
  548. {
  549. "Foo",
  550. new Item("Bar"),
  551. new TextBlock { Text = "Baz" },
  552. new ListBoxItem { Content = "Qux" },
  553. };
  554. var dataTemplate = new FuncDataTemplate<Item>((x, __) => new Button { Content = x });
  555. var target = CreateTarget(
  556. dataContext: "Base",
  557. itemsSource: items,
  558. itemTemplate: dataTemplate);
  559. var panel = Assert.IsAssignableFrom<Panel>(target.ItemsPanelRoot);
  560. var dataContexts = panel.Children
  561. .Do(x => (x as ContentPresenter)?.UpdateChild())
  562. .Cast<Control>()
  563. .Select(x => x.DataContext)
  564. .ToList();
  565. Assert.Equal(
  566. new object[] { items[0], items[1], "Base", "Base" },
  567. dataContexts);
  568. }
  569. [Fact]
  570. public void Control_Item_Should_Not_Be_NameScope()
  571. {
  572. using var app = Start();
  573. var items = new object[] { new TextBlock() };
  574. var target = CreateTarget(itemsSource: items);
  575. var item = target.LogicalChildren[0];
  576. Assert.Null(NameScope.GetNameScope((TextBlock)item));
  577. }
  578. [Fact]
  579. public void Focuses_Next_Item_On_Key_Down()
  580. {
  581. using var app = Start();
  582. var items = new object[]
  583. {
  584. new Button(),
  585. new Button(),
  586. };
  587. var target = CreateTarget(itemsSource: items);
  588. GetContainer<Button>(target).Focus();
  589. target.RaiseEvent(new KeyEventArgs
  590. {
  591. RoutedEvent = InputElement.KeyDownEvent,
  592. Key = Key.Down,
  593. });
  594. var panel = Assert.IsAssignableFrom<Panel>(target.ItemsPanelRoot);
  595. var focusManager = ((IInputRoot)target.VisualRoot!).FocusManager;
  596. Assert.Equal(panel.Children[1], focusManager?.GetFocusedElement());
  597. }
  598. [Fact]
  599. public void Does_Not_Focus_Non_Focusable_Item_On_Key_Down()
  600. {
  601. using var app = Start();
  602. var items = new object[]
  603. {
  604. new Button(),
  605. new Button { Focusable = false },
  606. new Button(),
  607. };
  608. var target = CreateTarget(itemsSource: items);
  609. GetContainer<Button>(target).Focus();
  610. target.RaiseEvent(new KeyEventArgs
  611. {
  612. RoutedEvent = InputElement.KeyDownEvent,
  613. Key = Key.Down,
  614. });
  615. var panel = Assert.IsAssignableFrom<Panel>(target.ItemsPanelRoot);
  616. var focusManager = ((IInputRoot)target.VisualRoot!).FocusManager;
  617. Assert.Equal(panel.Children[2], focusManager?.GetFocusedElement());
  618. }
  619. [Fact]
  620. public void Detaching_Then_Reattaching_To_Logical_Tree_Twice_Does_Not_Throw()
  621. {
  622. // # Issue 3487
  623. using var app = Start();
  624. var target = CreateTarget(
  625. itemsSource: new[] { "foo", "bar" },
  626. itemTemplate: new FuncDataTemplate<string>((_, __) => new Canvas()));
  627. var root = Assert.IsType<TestRoot>(target.GetVisualRoot());
  628. root.Child = null;
  629. root.Child = target;
  630. root.LayoutManager.ExecuteLayoutPass();
  631. root.Child = null;
  632. root.Child = target;
  633. }
  634. [Fact]
  635. public void Should_Use_DisplayMemberBinding()
  636. {
  637. using var app = Start();
  638. var target = CreateTarget(
  639. itemsSource: new[] { "Foo" },
  640. displayMemberBinding: new Binding("Length"));
  641. var container = GetContainer(target);
  642. var textBlock = Assert.IsType<TextBlock>(container.Child);
  643. Assert.Equal(textBlock.Text, "3");
  644. }
  645. [Fact]
  646. public void DisplayMemberBinding_Can_Be_Changed()
  647. {
  648. using var app = Start();
  649. var target = CreateTarget(
  650. itemsSource: new[] { new Item("Foo", "Bar") },
  651. displayMemberBinding: new Binding("Value"));
  652. var container = GetContainer(target);
  653. var textBlock = Assert.IsType<TextBlock>(container.Child);
  654. Assert.Equal(textBlock.Text, "Bar");
  655. target.DisplayMemberBinding = new Binding("Caption");
  656. Layout(target);
  657. container = GetContainer(target);
  658. textBlock = Assert.IsType<TextBlock>(container.Child);
  659. Assert.Equal(textBlock.Text, "Foo");
  660. }
  661. [Fact]
  662. public void Cannot_Set_Both_DisplayMemberBinding_And_ItemTemplate_1()
  663. {
  664. using var app = Start();
  665. var target = CreateTarget(
  666. displayMemberBinding: new Binding("Length"));
  667. Assert.Throws<InvalidOperationException>(() =>
  668. target.ItemTemplate = new FuncDataTemplate<string>((_, _) => new TextBlock()));
  669. }
  670. [Fact]
  671. public void Cannot_Set_Both_DisplayMemberBinding_And_ItemTemplate_2()
  672. {
  673. using var app = Start();
  674. var target = CreateTarget(
  675. itemTemplate: new FuncDataTemplate<string>((_, _) => new TextBlock()));
  676. Assert.Throws<InvalidOperationException>(() => target.DisplayMemberBinding = new Binding("Length"));
  677. }
  678. [Fact]
  679. public void ContainerPrepared_Is_Raised_For_Each_Control_Item_Container()
  680. {
  681. using var app = Start();
  682. var items = new AvaloniaList<string>();
  683. var target = CreateTarget();
  684. var result = new List<Control>();
  685. var index = 0;
  686. target.ContainerPrepared += (s, e) =>
  687. {
  688. Assert.Equal(index++, e.Index);
  689. result.Add(e.Container);
  690. };
  691. target.Items.Add(new Button());
  692. target.Items.Add(new Button());
  693. target.Items.Add(new Button());
  694. Assert.Equal(3, result.Count);
  695. Assert.Equal(target.GetRealizedContainers(), result);
  696. }
  697. [Fact]
  698. public void ContainerPrepared_Is_Raised_For_Each_Item_Container()
  699. {
  700. using var app = Start();
  701. var items = new AvaloniaList<string>();
  702. var target = CreateTarget();
  703. var result = new List<Control>();
  704. var index = 0;
  705. target.ContainerPrepared += (s, e) =>
  706. {
  707. Assert.Equal(index++, e.Index);
  708. result.Add(e.Container);
  709. };
  710. target.Items.Add("Foo");
  711. target.Items.Add("Bar");
  712. target.Items.Add("Baz");
  713. Assert.Equal(3, result.Count);
  714. Assert.Equal(target.GetRealizedContainers(), result);
  715. }
  716. [Fact]
  717. public void ContainerPrepared_Is_Raised_For_Each_ItemsSource_Item_Container_On_Layout()
  718. {
  719. using var app = Start();
  720. var items = new AvaloniaList<string>();
  721. var target = CreateTarget(itemsSource: items);
  722. var result = new List<Control>();
  723. var index = 0;
  724. target.ContainerPrepared += (s, e) =>
  725. {
  726. Assert.Equal(index++, e.Index);
  727. result.Add(e.Container);
  728. };
  729. items.AddRange(new[] { "Foo", "Bar", "Baz" });
  730. Assert.Equal(3, result.Count);
  731. Assert.Equal(target.GetRealizedContainers(), result);
  732. }
  733. [Fact]
  734. public void ContainerIndexChanged_Is_Raised_When_Item_Added()
  735. {
  736. using var app = Start();
  737. var target = CreateTarget(items: new[] { "Foo", "Bar", "Baz" });
  738. var result = new List<Control>();
  739. var index = 1;
  740. target.ContainerIndexChanged += (s, e) =>
  741. {
  742. Assert.Equal(index++, e.OldIndex);
  743. Assert.Equal(index, e.NewIndex);
  744. result.Add(e.Container);
  745. };
  746. target.Items.Insert(1, "Qux");
  747. Assert.Equal(2, result.Count);
  748. Assert.Equal(target.GetRealizedContainers().Skip(2), result);
  749. }
  750. [Fact]
  751. public void ContainerClearing_Is_Raised_When_Item_Removed()
  752. {
  753. using var app = Start();
  754. var target = CreateTarget(items: new[] { "Foo", "Bar", "Baz" });
  755. var expected = target.ContainerFromIndex(1);
  756. var raised = 0;
  757. target.ContainerClearing += (s, e) =>
  758. {
  759. Assert.Same(expected, e.Container);
  760. ++raised;
  761. };
  762. target.Items.RemoveAt(1);
  763. Assert.Equal(1, raised);
  764. }
  765. [Fact]
  766. public void ContainerClearing_Is_Raised_When_ItemsSource_Is_Cleared()
  767. {
  768. using var app = Start();
  769. var itemsSource = new ObservableCollection<object> { "Foo", "Bar", "Baz" };
  770. var target = CreateTarget(itemsSource: itemsSource);
  771. var expectedContainers = itemsSource.Select(x => target.ContainerFromItem(x)).ToArray();
  772. var actualContainers = new List<Control>();
  773. var raised = 0;
  774. target.ContainerClearing += (s, e) =>
  775. {
  776. actualContainers.Add(e.Container);
  777. ++raised;
  778. };
  779. itemsSource.Clear();
  780. Assert.Equal(3, raised);
  781. Assert.Equal(expectedContainers, actualContainers);
  782. }
  783. [Fact]
  784. public void Handles_Recycling_Control_Items_Inside_Containers()
  785. {
  786. // Issue #10825
  787. using var app = Start();
  788. // The items must be controls but not of the container type.
  789. var items = Enumerable.Range(0, 100).Select(x => new TextBlock
  790. {
  791. Text = $"Item {x}",
  792. Width = 100,
  793. Height = 100,
  794. }).ToList();
  795. // Virtualization is required
  796. var itemsPanel = new FuncTemplate<Panel?>(() => new VirtualizingStackPanel());
  797. // Create an ItemsControl which uses containers, and provide a scroll viewer.
  798. var target = CreateTarget<ItemsControlWithContainer>(
  799. items: items,
  800. itemsPanel: itemsPanel,
  801. scrollViewer: true);
  802. var scroll = target.FindAncestorOfType<ScrollViewer>();
  803. Assert.NotNull(scroll);
  804. Assert.Equal(10, target.GetRealizedContainers().Count());
  805. // Scroll so that half a container is visible: an extra container is generated.
  806. scroll.Offset = new(0, 2050);
  807. Layout(target);
  808. // Scroll so that the extra container is no longer needed and recycled.
  809. scroll.Offset = new(0, 2100);
  810. Layout(target);
  811. // Scroll back: issue #10825 triggered.
  812. scroll.Offset = new(0, 2000);
  813. Layout(target);
  814. }
  815. [Fact]
  816. public void ItemIsOwnContainer_Content_Should_Not_Be_Cleared_When_Removed()
  817. {
  818. // Issue #11128.
  819. using var app = Start();
  820. var item = new ContentPresenter { Content = "foo" };
  821. var target = CreateTarget(items: new[] { item });
  822. target.Items.RemoveAt(0);
  823. Assert.Equal("foo", item.Content);
  824. }
  825. private static ItemsControl CreateTarget(
  826. object? dataContext = null,
  827. IBinding? displayMemberBinding = null,
  828. IList? items = null,
  829. IList? itemsSource = null,
  830. ControlTheme? itemContainerTheme = null,
  831. IDataTemplate? itemTemplate = null,
  832. ITemplate<Panel?>? itemsPanel = null,
  833. IEnumerable<IDataTemplate>? dataTemplates = null,
  834. bool performLayout = true)
  835. {
  836. return CreateTarget<ItemsControl>(
  837. dataContext: dataContext,
  838. displayMemberBinding: displayMemberBinding,
  839. items: items,
  840. itemsSource: itemsSource,
  841. itemContainerTheme: itemContainerTheme,
  842. itemTemplate: itemTemplate,
  843. itemsPanel: itemsPanel,
  844. dataTemplates: dataTemplates,
  845. performLayout: performLayout);
  846. }
  847. private static T CreateTarget<T>(
  848. object? dataContext = null,
  849. IBinding? displayMemberBinding = null,
  850. IList? items = null,
  851. IList? itemsSource = null,
  852. ControlTheme? itemContainerTheme = null,
  853. IDataTemplate? itemTemplate = null,
  854. ITemplate<Panel?>? itemsPanel = null,
  855. IEnumerable<IDataTemplate>? dataTemplates = null,
  856. bool performLayout = true,
  857. bool scrollViewer = false)
  858. where T : ItemsControl, new()
  859. {
  860. var target = new T
  861. {
  862. DataContext = dataContext,
  863. DisplayMemberBinding = displayMemberBinding,
  864. ItemContainerTheme = itemContainerTheme,
  865. ItemTemplate = itemTemplate,
  866. ItemsSource = itemsSource,
  867. };
  868. if (items is not null)
  869. {
  870. foreach (var item in items)
  871. target.Items.Add(item);
  872. }
  873. if (itemsPanel is not null)
  874. target.ItemsPanel = itemsPanel;
  875. var scroll = scrollViewer ? new ScrollViewer { Content = target } : null;
  876. var root = CreateRoot(scroll ?? (Control)target);
  877. if (dataTemplates is not null)
  878. {
  879. foreach (var dataTemplate in dataTemplates)
  880. root.DataTemplates.Add(dataTemplate);
  881. }
  882. if (performLayout)
  883. root.LayoutManager.ExecuteInitialLayoutPass();
  884. return target;
  885. }
  886. private static TestRoot CreateRoot(Control child)
  887. {
  888. return new TestRoot
  889. {
  890. Resources =
  891. {
  892. { typeof(ContentControl), CreateContentControlTheme() },
  893. { typeof(ItemsControl), CreateItemsControlTheme() },
  894. { typeof(ScrollViewer), CreateScrollViewerTheme() },
  895. },
  896. Child = child,
  897. };
  898. }
  899. private static ControlTheme CreateContentControlTheme()
  900. {
  901. return new ControlTheme(typeof(ContentControl))
  902. {
  903. Setters =
  904. {
  905. new Setter(TreeView.TemplateProperty, CreateContentControlTemplate()),
  906. },
  907. };
  908. }
  909. private static FuncControlTemplate CreateContentControlTemplate()
  910. {
  911. return new FuncControlTemplate<ContentControl>((parent, scope) =>
  912. new ContentPresenter
  913. {
  914. Name = "PART_ContentPresenter",
  915. [!ContentPresenter.ContentProperty] = parent[!ListBoxItem.ContentProperty],
  916. [!ContentPresenter.ContentTemplateProperty] = parent[!ListBoxItem.ContentTemplateProperty],
  917. }.RegisterInNameScope(scope));
  918. }
  919. private static ControlTheme CreateItemsControlTheme()
  920. {
  921. return new ControlTheme(typeof(ItemsControl))
  922. {
  923. Setters =
  924. {
  925. new Setter(TreeView.TemplateProperty, CreateItemsControlTemplate()),
  926. },
  927. };
  928. }
  929. private static FuncControlTemplate CreateItemsControlTemplate()
  930. {
  931. return new FuncControlTemplate<ItemsControl>((parent, scope) =>
  932. {
  933. return new Border
  934. {
  935. Background = new Media.SolidColorBrush(0xffffffff),
  936. Child = new ItemsPresenter
  937. {
  938. Name = "PART_ItemsPresenter",
  939. [~ItemsPresenter.ItemsPanelProperty] = parent[~ItemsControl.ItemsPanelProperty],
  940. }.RegisterInNameScope(scope)
  941. };
  942. });
  943. }
  944. private static ControlTheme CreateScrollViewerTheme()
  945. {
  946. return new ControlTheme(typeof(ScrollViewer))
  947. {
  948. Setters =
  949. {
  950. new Setter(TreeView.TemplateProperty, CreateScrollViewerTemplate()),
  951. },
  952. };
  953. }
  954. private static FuncControlTemplate CreateScrollViewerTemplate()
  955. {
  956. return new FuncControlTemplate<ScrollViewer>((parent, scope) =>
  957. new Panel
  958. {
  959. Children =
  960. {
  961. new ScrollContentPresenter
  962. {
  963. Name = "PART_ContentPresenter",
  964. }.RegisterInNameScope(scope),
  965. new ScrollBar
  966. {
  967. Name = "verticalScrollBar",
  968. }
  969. }
  970. });
  971. }
  972. private static void Layout(Control c)
  973. {
  974. (c.GetVisualRoot() as ILayoutRoot)?.LayoutManager.ExecuteLayoutPass();
  975. }
  976. private static ContentPresenter GetContainer(ItemsControl target, int index = 0)
  977. {
  978. return Assert.IsType<ContentPresenter>(target.GetRealizedContainers().ElementAt(index));
  979. }
  980. private static T GetContainer<T>(ItemsControl target, int index = 0)
  981. {
  982. return Assert.IsType<T>(target.GetRealizedContainers().ElementAt(index));
  983. }
  984. public static IDisposable Start()
  985. {
  986. return UnitTestApplication.Start(
  987. TestServices.MockThreadingInterface.With(
  988. fontManagerImpl: new HeadlessFontManagerStub(),
  989. keyboardDevice: () => new KeyboardDevice(),
  990. keyboardNavigation: () => new KeyboardNavigationHandler(),
  991. inputManager: new InputManager(),
  992. renderInterface: new HeadlessPlatformRenderInterface(),
  993. textShaperImpl: new HeadlessTextShaperStub()));
  994. }
  995. private class ItemsControlWithContainer : ItemsControl
  996. {
  997. protected override Type StyleKeyOverride => typeof(ItemsControl);
  998. protected internal override Control CreateContainerForItemOverride(object? item, int index, object? recycleKey)
  999. {
  1000. return new ContainerControl();
  1001. }
  1002. protected internal override bool NeedsContainerOverride(object? item, int index, out object? recycleKey)
  1003. {
  1004. return NeedsContainer<ContainerControl>(item, out recycleKey);
  1005. }
  1006. }
  1007. private class ContainerControl : ContentControl
  1008. {
  1009. protected override Type StyleKeyOverride => typeof(ContentControl);
  1010. }
  1011. private record Item(string Caption, string? Value = null);
  1012. }
  1013. }