ItemsPresenterTests_Virtualization_Simple.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598
  1. // Copyright (c) The Avalonia Project. All rights reserved.
  2. // Licensed under the MIT license. See licence.md file in the project root for full license information.
  3. using System.Collections;
  4. using System.Collections.Generic;
  5. using System.Collections.ObjectModel;
  6. using System.Linq;
  7. using Avalonia.Collections;
  8. using Avalonia.Controls.Generators;
  9. using Avalonia.Controls.Presenters;
  10. using Avalonia.Controls.Primitives;
  11. using Avalonia.Controls.Templates;
  12. using Avalonia.Input;
  13. using Xunit;
  14. namespace Avalonia.Controls.UnitTests.Presenters
  15. {
  16. public class ItemsPresenterTests_Virtualization_Simple
  17. {
  18. [Fact]
  19. public void Should_Return_Items_Count_For_Extent_Vertical()
  20. {
  21. var target = CreateTarget();
  22. target.ApplyTemplate();
  23. Assert.Equal(new Size(0, 20), ((ILogicalScrollable)target).Extent);
  24. }
  25. [Fact]
  26. public void Should_Return_Items_Count_For_Extent_Horizontal()
  27. {
  28. var target = CreateTarget(orientation: Orientation.Horizontal);
  29. target.ApplyTemplate();
  30. Assert.Equal(new Size(20, 0), ((ILogicalScrollable)target).Extent);
  31. }
  32. [Fact]
  33. public void Should_Have_Number_Of_Visible_Items_As_Viewport_Vertical()
  34. {
  35. var target = CreateTarget();
  36. target.ApplyTemplate();
  37. target.Measure(new Size(100, 100));
  38. target.Arrange(new Rect(0, 0, 100, 100));
  39. Assert.Equal(new Size(0, 10), ((ILogicalScrollable)target).Viewport);
  40. }
  41. [Fact]
  42. public void Should_Have_Number_Of_Visible_Items_As_Viewport_Horizontal()
  43. {
  44. var target = CreateTarget(orientation: Orientation.Horizontal);
  45. target.ApplyTemplate();
  46. target.Measure(new Size(100, 100));
  47. target.Arrange(new Rect(0, 0, 100, 100));
  48. Assert.Equal(new Size(10, 0), ((ILogicalScrollable)target).Viewport);
  49. }
  50. [Fact]
  51. public void Should_Remove_Items_When_Control_Is_Shrank()
  52. {
  53. var target = CreateTarget();
  54. var items = (IList<string>)target.Items;
  55. target.ApplyTemplate();
  56. target.Measure(new Size(100, 100));
  57. target.Arrange(new Rect(0, 0, 100, 100));
  58. Assert.Equal(10, target.Panel.Children.Count);
  59. target.Measure(new Size(100, 80));
  60. target.Arrange(new Rect(0, 0, 100, 80));
  61. Assert.Equal(8, target.Panel.Children.Count);
  62. }
  63. [Fact]
  64. public void Should_Add_New_Containers_At_Top_When_Control_Is_Scrolled_To_Bottom_And_Enlarged()
  65. {
  66. var target = CreateTarget();
  67. var items = (IList<string>)target.Items;
  68. target.ApplyTemplate();
  69. target.Measure(new Size(100, 100));
  70. target.Arrange(new Rect(0, 0, 100, 100));
  71. Assert.Equal(10, target.Panel.Children.Count);
  72. ((IScrollable)target).Offset = new Vector(0, 10);
  73. target.Measure(new Size(120, 120));
  74. target.Arrange(new Rect(0, 0, 100, 120));
  75. Assert.Equal(12, target.Panel.Children.Count);
  76. for (var i = 0; i < target.Panel.Children.Count; ++i)
  77. {
  78. Assert.Equal(items[i + 8], target.Panel.Children[i].DataContext);
  79. }
  80. }
  81. [Fact]
  82. public void Should_Update_Containers_When_Items_Changes()
  83. {
  84. var target = CreateTarget();
  85. target.ApplyTemplate();
  86. target.Measure(new Size(100, 100));
  87. target.Arrange(new Rect(0, 0, 100, 100));
  88. target.Items = new[] { "foo", "bar", "baz" };
  89. Assert.Equal(3, target.Panel.Children.Count);
  90. }
  91. [Fact]
  92. public void Should_Decrease_The_Viewport_Size_By_One_If_There_Is_A_Partial_Item()
  93. {
  94. var target = CreateTarget();
  95. target.ApplyTemplate();
  96. target.Measure(new Size(100, 95));
  97. target.Arrange(new Rect(0, 0, 100, 95));
  98. Assert.Equal(new Size(0, 9), ((ILogicalScrollable)target).Viewport);
  99. }
  100. [Fact]
  101. public void Moving_To_And_From_The_End_With_Partial_Item_Should_Set_Panel_PixelOffset()
  102. {
  103. var target = CreateTarget();
  104. target.ApplyTemplate();
  105. target.Measure(new Size(100, 95));
  106. target.Arrange(new Rect(0, 0, 100, 95));
  107. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  108. var minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  109. Assert.Equal(new Vector(0, 11), ((ILogicalScrollable)target).Offset);
  110. Assert.Equal(10, minIndex);
  111. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  112. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  113. minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  114. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  115. Assert.Equal(10, minIndex);
  116. Assert.Equal(0, ((IVirtualizingPanel)target.Panel).PixelOffset);
  117. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  118. minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  119. Assert.Equal(new Vector(0, 11), ((ILogicalScrollable)target).Offset);
  120. Assert.Equal(10, minIndex);
  121. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  122. }
  123. [Fact]
  124. public void Inserting_Items_Should_Update_Containers()
  125. {
  126. var target = CreateTarget();
  127. target.ApplyTemplate();
  128. target.Measure(new Size(100, 100));
  129. target.Arrange(new Rect(0, 0, 100, 100));
  130. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  131. var expected = Enumerable.Range(5, 10).Select(x => $"Item {x}").ToList();
  132. var items = (ObservableCollection<string>)target.Items;
  133. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  134. Assert.Equal(expected, actual);
  135. items.Insert(6, "Inserted");
  136. expected.Insert(1, "Inserted");
  137. expected.RemoveAt(expected.Count - 1);
  138. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  139. Assert.Equal(expected, actual);
  140. }
  141. [Fact]
  142. public void Removing_First_Materialized_Item_Should_Update_Containers()
  143. {
  144. var target = CreateTarget();
  145. target.ApplyTemplate();
  146. target.Measure(new Size(100, 100));
  147. target.Arrange(new Rect(0, 0, 100, 100));
  148. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  149. var items = (ObservableCollection<string>)target.Items;
  150. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  151. Assert.Equal(expected, actual);
  152. items.RemoveAt(0);
  153. expected = Enumerable.Range(1, 10).Select(x => $"Item {x}").ToList();
  154. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  155. Assert.Equal(expected, actual);
  156. }
  157. [Fact]
  158. public void Removing_Items_From_Middle_Should_Update_Containers_When_All_Items_Visible()
  159. {
  160. var target = CreateTarget();
  161. target.ApplyTemplate();
  162. target.Measure(new Size(100, 200));
  163. target.Arrange(new Rect(0, 0, 100, 200));
  164. var items = (ObservableCollection<string>)target.Items;
  165. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  166. Assert.Equal(items, actual);
  167. items.RemoveAt(2);
  168. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  169. Assert.Equal(items, actual);
  170. items.RemoveAt(items.Count - 2);
  171. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  172. Assert.Equal(items, actual);
  173. }
  174. [Fact]
  175. public void Removing_Last_Item_Should_Update_Containers_When_All_Items_Visible()
  176. {
  177. var target = CreateTarget();
  178. target.ApplyTemplate();
  179. target.Measure(new Size(100, 200));
  180. target.Arrange(new Rect(0, 0, 100, 200));
  181. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  182. var expected = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList();
  183. var items = (ObservableCollection<string>)target.Items;
  184. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  185. Assert.Equal(expected, actual);
  186. items.Remove(items.Last());
  187. expected.Remove(expected.Last());
  188. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  189. Assert.Equal(expected, actual);
  190. }
  191. [Fact]
  192. public void Removing_Items_When_Scrolled_To_End_Should_Recyle_Containers_At_Top()
  193. {
  194. var target = CreateTarget(useAvaloniaList: true);
  195. target.ApplyTemplate();
  196. target.Measure(new Size(100, 100));
  197. target.Arrange(new Rect(0, 0, 100, 100));
  198. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  199. var expected = Enumerable.Range(10, 10).Select(x => $"Item {x}").ToList();
  200. var items = (AvaloniaList<string>)target.Items;
  201. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  202. Assert.Equal(expected, actual);
  203. items.RemoveRange(18, 2);
  204. expected = Enumerable.Range(8, 10).Select(x => $"Item {x}").ToList();
  205. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  206. Assert.Equal(expected, actual);
  207. }
  208. [Fact]
  209. public void Removing_Items_When_Scrolled_To_Near_End_Should_Recycle_Containers_At_Bottom_And_Top()
  210. {
  211. var target = CreateTarget(useAvaloniaList: true);
  212. target.ApplyTemplate();
  213. target.Measure(new Size(100, 100));
  214. target.Arrange(new Rect(0, 0, 100, 100));
  215. ((ILogicalScrollable)target).Offset = new Vector(0, 9);
  216. var expected = Enumerable.Range(9, 10).Select(x => $"Item {x}").ToList();
  217. var items = (AvaloniaList<string>)target.Items;
  218. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  219. Assert.Equal(expected, actual);
  220. items.RemoveRange(15, 3);
  221. expected = Enumerable.Range(7, 8).Select(x => $"Item {x}")
  222. .Concat(Enumerable.Range(18, 2).Select(x => $"Item {x}"))
  223. .ToList();
  224. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  225. Assert.Equal(expected, actual);
  226. }
  227. [Fact]
  228. public void Replacing_Items_Should_Update_Containers()
  229. {
  230. var target = CreateTarget();
  231. target.ApplyTemplate();
  232. target.Measure(new Size(100, 100));
  233. target.Arrange(new Rect(0, 0, 100, 100));
  234. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  235. var items = (ObservableCollection<string>)target.Items;
  236. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  237. Assert.Equal(expected, actual);
  238. items[4] = expected[4] = "Replaced";
  239. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  240. Assert.Equal(expected, actual);
  241. }
  242. [Fact]
  243. public void Moving_Items_Should_Update_Containers()
  244. {
  245. var target = CreateTarget();
  246. target.ApplyTemplate();
  247. target.Measure(new Size(100, 100));
  248. target.Arrange(new Rect(0, 0, 100, 100));
  249. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  250. var items = (ObservableCollection<string>)target.Items;
  251. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  252. Assert.Equal(expected, actual);
  253. items.Move(4, 8);
  254. var i = expected[4];
  255. expected.RemoveAt(4);
  256. expected.Insert(8, i);
  257. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  258. Assert.Equal(expected, actual);
  259. }
  260. [Fact]
  261. public void Setting_Items_To_Null_Should_Remove_Containers()
  262. {
  263. var target = CreateTarget();
  264. target.ApplyTemplate();
  265. target.Measure(new Size(100, 100));
  266. target.Arrange(new Rect(0, 0, 100, 100));
  267. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  268. var items = (ObservableCollection<string>)target.Items;
  269. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  270. Assert.Equal(expected, actual);
  271. target.Items = null;
  272. Assert.Empty(target.Panel.Children);
  273. }
  274. public class Vertical
  275. {
  276. [Fact]
  277. public void GetControlInDirection_Down_Should_Return_Existing_Container_If_Materialized()
  278. {
  279. var target = CreateTarget();
  280. target.ApplyTemplate();
  281. target.Measure(new Size(100, 100));
  282. target.Arrange(new Rect(0, 0, 100, 100));
  283. var from = target.Panel.Children[5];
  284. var result = ((ILogicalScrollable)target).GetControlInDirection(
  285. FocusNavigationDirection.Down,
  286. from);
  287. Assert.Same(target.Panel.Children[6], result);
  288. }
  289. [Fact]
  290. public void GetControlInDirection_Down_Should_Scroll_If_Necessary()
  291. {
  292. var target = CreateTarget();
  293. target.ApplyTemplate();
  294. target.Measure(new Size(100, 100));
  295. target.Arrange(new Rect(0, 0, 100, 100));
  296. var from = target.Panel.Children[9];
  297. var result = ((ILogicalScrollable)target).GetControlInDirection(
  298. FocusNavigationDirection.Down,
  299. from);
  300. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  301. Assert.Same(target.Panel.Children[9], result);
  302. }
  303. [Fact]
  304. public void GetControlInDirection_Down_Should_Scroll_If_Partially_Visible()
  305. {
  306. var target = CreateTarget();
  307. target.ApplyTemplate();
  308. target.Measure(new Size(100, 95));
  309. target.Arrange(new Rect(0, 0, 100, 95));
  310. var from = target.Panel.Children[8];
  311. var result = ((ILogicalScrollable)target).GetControlInDirection(
  312. FocusNavigationDirection.Down,
  313. from);
  314. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  315. Assert.Same(target.Panel.Children[8], result);
  316. }
  317. [Fact]
  318. public void GetControlInDirection_Up_Should_Scroll_If_Partially_Visible_Is_Currently_Shown()
  319. {
  320. var target = CreateTarget();
  321. target.ApplyTemplate();
  322. target.Measure(new Size(100, 95));
  323. target.Arrange(new Rect(0, 0, 100, 95));
  324. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  325. var from = target.Panel.Children[1];
  326. var result = ((ILogicalScrollable)target).GetControlInDirection(
  327. FocusNavigationDirection.Up,
  328. from);
  329. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  330. Assert.Same(target.Panel.Children[0], result);
  331. }
  332. }
  333. public class WithContainers
  334. {
  335. [Fact]
  336. public void Scrolling_Less_Than_A_Page_Should_Move_Recycled_Items()
  337. {
  338. var target = CreateTarget();
  339. var items = (IList<string>)target.Items;
  340. target.ApplyTemplate();
  341. target.Measure(new Size(100, 100));
  342. target.Arrange(new Rect(0, 0, 100, 100));
  343. var containers = target.Panel.Children.ToList();
  344. var scroller = (ScrollContentPresenter)target.Parent;
  345. scroller.Offset = new Vector(0, 5);
  346. var scrolledContainers = containers
  347. .Skip(5)
  348. .Take(5)
  349. .Concat(containers.Take(5)).ToList();
  350. Assert.Equal(new Vector(0, 5), ((ILogicalScrollable)target).Offset);
  351. Assert.Equal(scrolledContainers, target.Panel.Children);
  352. for (var i = 0; i < target.Panel.Children.Count; ++i)
  353. {
  354. Assert.Equal(items[i + 5], target.Panel.Children[i].DataContext);
  355. }
  356. scroller.Offset = new Vector(0, 0);
  357. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  358. Assert.Equal(containers, target.Panel.Children);
  359. var dcs = target.Panel.Children.Select(x => x.DataContext).ToList();
  360. for (var i = 0; i < target.Panel.Children.Count; ++i)
  361. {
  362. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  363. }
  364. }
  365. [Fact]
  366. public void Scrolling_More_Than_A_Page_Should_Recycle_Items()
  367. {
  368. var target = CreateTarget(itemCount: 50);
  369. var items = (IList<string>)target.Items;
  370. target.ApplyTemplate();
  371. target.Measure(new Size(100, 100));
  372. target.Arrange(new Rect(0, 0, 100, 100));
  373. var containers = target.Panel.Children.ToList();
  374. var scroller = (ScrollContentPresenter)target.Parent;
  375. scroller.Offset = new Vector(0, 20);
  376. Assert.Equal(new Vector(0, 20), ((ILogicalScrollable)target).Offset);
  377. Assert.Equal(containers, target.Panel.Children);
  378. for (var i = 0; i < target.Panel.Children.Count; ++i)
  379. {
  380. Assert.Equal(items[i + 20], target.Panel.Children[i].DataContext);
  381. }
  382. scroller.Offset = new Vector(0, 0);
  383. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  384. Assert.Equal(containers, target.Panel.Children);
  385. for (var i = 0; i < target.Panel.Children.Count; ++i)
  386. {
  387. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  388. }
  389. }
  390. }
  391. private static ItemsPresenter CreateTarget(
  392. Orientation orientation = Orientation.Vertical,
  393. bool useContainers = true,
  394. int itemCount = 20,
  395. bool useAvaloniaList = false)
  396. {
  397. ItemsPresenter result;
  398. var itemsSource = Enumerable.Range(0, itemCount).Select(x => $"Item {x}");
  399. var items = useAvaloniaList ?
  400. (IEnumerable)new AvaloniaList<string>(itemsSource) :
  401. (IEnumerable)new ObservableCollection<string>(itemsSource);
  402. var scroller = new ScrollContentPresenter
  403. {
  404. Content = result = new TestItemsPresenter(useContainers)
  405. {
  406. Items = items,
  407. ItemsPanel = VirtualizingPanelTemplate(orientation),
  408. ItemTemplate = ItemTemplate(),
  409. VirtualizationMode = ItemVirtualizationMode.Simple,
  410. }
  411. };
  412. scroller.UpdateChild();
  413. return result;
  414. }
  415. private static IDataTemplate ItemTemplate()
  416. {
  417. return new FuncDataTemplate<string>(x => new Canvas
  418. {
  419. Width = 10,
  420. Height = 10,
  421. });
  422. }
  423. private static ITemplate<IPanel> VirtualizingPanelTemplate(
  424. Orientation orientation = Orientation.Vertical)
  425. {
  426. return new FuncTemplate<IPanel>(() => new VirtualizingStackPanel
  427. {
  428. Orientation = orientation,
  429. });
  430. }
  431. private class TestItemsPresenter : ItemsPresenter
  432. {
  433. private bool _useContainers;
  434. public TestItemsPresenter(bool useContainers)
  435. {
  436. _useContainers = useContainers;
  437. }
  438. protected override IItemContainerGenerator CreateItemContainerGenerator()
  439. {
  440. return _useContainers ?
  441. new ItemContainerGenerator<TestContainer>(this, TestContainer.ContentProperty, null) :
  442. new ItemContainerGenerator(this);
  443. }
  444. }
  445. private class TestContainer : ContentControl
  446. {
  447. public TestContainer()
  448. {
  449. Width = 10;
  450. Height = 10;
  451. }
  452. }
  453. }
  454. }