ItemsPresenterTests_Virtualization_Simple.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831
  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;
  4. using System.Collections;
  5. using System.Collections.Generic;
  6. using System.Collections.ObjectModel;
  7. using System.Linq;
  8. using Avalonia.Collections;
  9. using Avalonia.Controls.Generators;
  10. using Avalonia.Controls.Presenters;
  11. using Avalonia.Controls.Primitives;
  12. using Avalonia.Controls.Templates;
  13. using Avalonia.Input;
  14. using Avalonia.Rendering;
  15. using Avalonia.UnitTests;
  16. using Xunit;
  17. namespace Avalonia.Controls.UnitTests.Presenters
  18. {
  19. public class ItemsPresenterTests_Virtualization_Simple
  20. {
  21. [Fact]
  22. public void Should_Return_Items_Count_For_Extent_Vertical()
  23. {
  24. var target = CreateTarget();
  25. target.ApplyTemplate();
  26. Assert.Equal(new Size(0, 20), ((ILogicalScrollable)target).Extent);
  27. }
  28. [Fact]
  29. public void Should_Return_Items_Count_For_Extent_Horizontal()
  30. {
  31. var target = CreateTarget(orientation: Orientation.Horizontal);
  32. target.ApplyTemplate();
  33. Assert.Equal(new Size(20, 0), ((ILogicalScrollable)target).Extent);
  34. }
  35. [Fact]
  36. public void Should_Have_Number_Of_Visible_Items_As_Viewport_Vertical()
  37. {
  38. var target = CreateTarget();
  39. target.ApplyTemplate();
  40. target.Measure(new Size(100, 100));
  41. target.Arrange(new Rect(0, 0, 100, 100));
  42. Assert.Equal(new Size(0, 10), ((ILogicalScrollable)target).Viewport);
  43. }
  44. [Fact]
  45. public void Should_Have_Number_Of_Visible_Items_As_Viewport_Horizontal()
  46. {
  47. var target = CreateTarget(orientation: Orientation.Horizontal);
  48. target.ApplyTemplate();
  49. target.Measure(new Size(100, 100));
  50. target.Arrange(new Rect(0, 0, 100, 100));
  51. Assert.Equal(new Size(10, 0), ((ILogicalScrollable)target).Viewport);
  52. }
  53. [Fact]
  54. public void Should_Add_Containers_When_Panel_Is_Not_Full()
  55. {
  56. var target = CreateTarget(itemCount: 5);
  57. var items = (IList<string>)target.Items;
  58. target.ApplyTemplate();
  59. target.Measure(new Size(100, 100));
  60. target.Arrange(new Rect(0, 0, 100, 100));
  61. Assert.Equal(5, target.Panel.Children.Count);
  62. items.Add("New Item");
  63. Assert.Equal(6, target.Panel.Children.Count);
  64. }
  65. [Fact]
  66. public void Should_Remove_Items_When_Control_Is_Shrank()
  67. {
  68. var target = CreateTarget();
  69. var items = (IList<string>)target.Items;
  70. target.ApplyTemplate();
  71. target.Measure(new Size(100, 100));
  72. target.Arrange(new Rect(0, 0, 100, 100));
  73. Assert.Equal(10, target.Panel.Children.Count);
  74. target.Measure(new Size(100, 80));
  75. target.Arrange(new Rect(0, 0, 100, 80));
  76. Assert.Equal(8, target.Panel.Children.Count);
  77. }
  78. [Fact]
  79. public void Should_Add_New_Containers_At_Top_When_Control_Is_Scrolled_To_Bottom_And_Enlarged()
  80. {
  81. var target = CreateTarget();
  82. var items = (IList<string>)target.Items;
  83. target.ApplyTemplate();
  84. target.Measure(new Size(100, 100));
  85. target.Arrange(new Rect(0, 0, 100, 100));
  86. Assert.Equal(10, target.Panel.Children.Count);
  87. ((IScrollable)target).Offset = new Vector(0, 10);
  88. target.Measure(new Size(120, 120));
  89. target.Arrange(new Rect(0, 0, 100, 120));
  90. Assert.Equal(12, target.Panel.Children.Count);
  91. for (var i = 0; i < target.Panel.Children.Count; ++i)
  92. {
  93. Assert.Equal(items[i + 8], target.Panel.Children[i].DataContext);
  94. }
  95. }
  96. [Fact]
  97. public void Should_Update_Containers_When_Items_Changes()
  98. {
  99. var target = CreateTarget();
  100. target.ApplyTemplate();
  101. target.Measure(new Size(100, 100));
  102. target.Arrange(new Rect(0, 0, 100, 100));
  103. target.Items = new[] { "foo", "bar", "baz" };
  104. Assert.Equal(3, target.Panel.Children.Count);
  105. }
  106. [Fact]
  107. public void Should_Decrease_The_Viewport_Size_By_One_If_There_Is_A_Partial_Item()
  108. {
  109. var target = CreateTarget();
  110. target.ApplyTemplate();
  111. target.Measure(new Size(100, 95));
  112. target.Arrange(new Rect(0, 0, 100, 95));
  113. Assert.Equal(new Size(0, 9), ((ILogicalScrollable)target).Viewport);
  114. }
  115. [Fact]
  116. public void Moving_To_And_From_The_End_With_Partial_Item_Should_Set_Panel_PixelOffset()
  117. {
  118. var target = CreateTarget();
  119. target.ApplyTemplate();
  120. target.Measure(new Size(100, 95));
  121. target.Arrange(new Rect(0, 0, 100, 95));
  122. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  123. var minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  124. Assert.Equal(new Vector(0, 11), ((ILogicalScrollable)target).Offset);
  125. Assert.Equal(10, minIndex);
  126. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  127. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  128. minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  129. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  130. Assert.Equal(10, minIndex);
  131. Assert.Equal(0, ((IVirtualizingPanel)target.Panel).PixelOffset);
  132. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  133. minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  134. Assert.Equal(new Vector(0, 11), ((ILogicalScrollable)target).Offset);
  135. Assert.Equal(10, minIndex);
  136. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  137. }
  138. [Fact]
  139. public void Inserting_Items_Should_Update_Containers()
  140. {
  141. var target = CreateTarget();
  142. target.ApplyTemplate();
  143. target.Measure(new Size(100, 100));
  144. target.Arrange(new Rect(0, 0, 100, 100));
  145. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  146. var expected = Enumerable.Range(5, 10).Select(x => $"Item {x}").ToList();
  147. var items = (ObservableCollection<string>)target.Items;
  148. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  149. Assert.Equal(expected, actual);
  150. items.Insert(6, "Inserted");
  151. expected.Insert(1, "Inserted");
  152. expected.RemoveAt(expected.Count - 1);
  153. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  154. Assert.Equal(expected, actual);
  155. }
  156. [Fact]
  157. public void Removing_First_Materialized_Item_Should_Update_Containers()
  158. {
  159. var target = CreateTarget();
  160. target.ApplyTemplate();
  161. target.Measure(new Size(100, 100));
  162. target.Arrange(new Rect(0, 0, 100, 100));
  163. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  164. var items = (ObservableCollection<string>)target.Items;
  165. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  166. Assert.Equal(expected, actual);
  167. items.RemoveAt(0);
  168. expected = Enumerable.Range(1, 10).Select(x => $"Item {x}").ToList();
  169. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  170. Assert.Equal(expected, actual);
  171. }
  172. [Fact]
  173. public void Removing_Items_From_Middle_Should_Update_Containers_When_All_Items_Visible()
  174. {
  175. var target = CreateTarget();
  176. target.ApplyTemplate();
  177. target.Measure(new Size(100, 200));
  178. target.Arrange(new Rect(0, 0, 100, 200));
  179. var items = (ObservableCollection<string>)target.Items;
  180. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  181. Assert.Equal(items, actual);
  182. items.RemoveAt(2);
  183. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  184. Assert.Equal(items, actual);
  185. items.RemoveAt(items.Count - 2);
  186. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  187. Assert.Equal(items, actual);
  188. }
  189. [Fact]
  190. public void Removing_Last_Item_Should_Update_Containers_When_All_Items_Visible()
  191. {
  192. var target = CreateTarget();
  193. target.ApplyTemplate();
  194. target.Measure(new Size(100, 200));
  195. target.Arrange(new Rect(0, 0, 100, 200));
  196. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  197. var expected = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList();
  198. var items = (ObservableCollection<string>)target.Items;
  199. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  200. Assert.Equal(expected, actual);
  201. items.Remove(items.Last());
  202. expected.Remove(expected.Last());
  203. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  204. Assert.Equal(expected, actual);
  205. }
  206. [Fact]
  207. public void Removing_Items_When_Scrolled_To_End_Should_Recyle_Containers_At_Top()
  208. {
  209. var target = CreateTarget(useAvaloniaList: true);
  210. target.ApplyTemplate();
  211. target.Measure(new Size(100, 100));
  212. target.Arrange(new Rect(0, 0, 100, 100));
  213. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  214. var expected = Enumerable.Range(10, 10).Select(x => $"Item {x}").ToList();
  215. var items = (AvaloniaList<string>)target.Items;
  216. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  217. Assert.Equal(expected, actual);
  218. items.RemoveRange(18, 2);
  219. expected = Enumerable.Range(8, 10).Select(x => $"Item {x}").ToList();
  220. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  221. Assert.Equal(expected, actual);
  222. }
  223. [Fact]
  224. public void Removing_Items_When_Scrolled_To_Near_End_Should_Recycle_Containers_At_Bottom_And_Top()
  225. {
  226. var target = CreateTarget(useAvaloniaList: true);
  227. target.ApplyTemplate();
  228. target.Measure(new Size(100, 100));
  229. target.Arrange(new Rect(0, 0, 100, 100));
  230. ((ILogicalScrollable)target).Offset = new Vector(0, 9);
  231. var expected = Enumerable.Range(9, 10).Select(x => $"Item {x}").ToList();
  232. var items = (AvaloniaList<string>)target.Items;
  233. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  234. Assert.Equal(expected, actual);
  235. items.RemoveRange(15, 3);
  236. expected = Enumerable.Range(7, 8).Select(x => $"Item {x}")
  237. .Concat(Enumerable.Range(18, 2).Select(x => $"Item {x}"))
  238. .ToList();
  239. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  240. Assert.Equal(expected, actual);
  241. }
  242. [Fact]
  243. public void Measuring_To_Infinity_When_Scrolled_To_End_Should_Not_Throw()
  244. {
  245. var target = CreateTarget(useAvaloniaList: true);
  246. target.ApplyTemplate();
  247. target.Measure(new Size(100, 100));
  248. target.Arrange(new Rect(0, 0, 100, 100));
  249. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  250. // Check for issue #589: this should not throw.
  251. target.Measure(Size.Infinity);
  252. var expected = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList();
  253. var items = (AvaloniaList<string>)target.Items;
  254. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  255. Assert.Equal(expected, actual);
  256. }
  257. [Fact]
  258. public void Replacing_Items_Should_Update_Containers()
  259. {
  260. var target = CreateTarget();
  261. target.ApplyTemplate();
  262. target.Measure(new Size(100, 100));
  263. target.Arrange(new Rect(0, 0, 100, 100));
  264. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  265. var items = (ObservableCollection<string>)target.Items;
  266. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  267. Assert.Equal(expected, actual);
  268. items[4] = expected[4] = "Replaced";
  269. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  270. Assert.Equal(expected, actual);
  271. }
  272. [Fact]
  273. public void Moving_Items_Should_Update_Containers()
  274. {
  275. var target = CreateTarget();
  276. target.ApplyTemplate();
  277. target.Measure(new Size(100, 100));
  278. target.Arrange(new Rect(0, 0, 100, 100));
  279. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  280. var items = (ObservableCollection<string>)target.Items;
  281. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  282. Assert.Equal(expected, actual);
  283. items.Move(4, 8);
  284. var i = expected[4];
  285. expected.RemoveAt(4);
  286. expected.Insert(8, i);
  287. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  288. Assert.Equal(expected, actual);
  289. }
  290. [Fact]
  291. public void Setting_Items_To_Null_Should_Remove_Containers()
  292. {
  293. var target = CreateTarget();
  294. target.ApplyTemplate();
  295. target.Measure(new Size(100, 100));
  296. target.Arrange(new Rect(0, 0, 100, 100));
  297. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  298. var items = (ObservableCollection<string>)target.Items;
  299. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  300. Assert.Equal(expected, actual);
  301. target.Items = null;
  302. Assert.Empty(target.Panel.Children);
  303. }
  304. [Fact]
  305. public void Reassigning_Items_Should_Create_Containers()
  306. {
  307. var target = CreateTarget(itemCount: 5);
  308. target.ApplyTemplate();
  309. target.Measure(new Size(100, 100));
  310. target.Arrange(new Rect(0, 0, 100, 100));
  311. var expected = Enumerable.Range(0, 5).Select(x => $"Item {x}").ToList();
  312. var items = (ObservableCollection<string>)target.Items;
  313. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  314. Assert.Equal(expected, actual);
  315. expected = Enumerable.Range(0, 6).Select(x => $"Item {x}").ToList();
  316. target.Items = expected;
  317. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  318. Assert.Equal(expected, actual);
  319. }
  320. [Fact]
  321. public void Reassigning_Items_Should_Remove_Containers()
  322. {
  323. var target = CreateTarget(itemCount: 6);
  324. target.ApplyTemplate();
  325. target.Measure(new Size(100, 100));
  326. target.Arrange(new Rect(0, 0, 100, 100));
  327. var expected = Enumerable.Range(0, 6).Select(x => $"Item {x}").ToList();
  328. var items = (ObservableCollection<string>)target.Items;
  329. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  330. Assert.Equal(expected, actual);
  331. expected = Enumerable.Range(0, 5).Select(x => $"Item {x}").ToList();
  332. target.Items = expected;
  333. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  334. Assert.Equal(expected, actual);
  335. }
  336. [Fact]
  337. public void Clearing_Items_And_ReAdding_Should_Remove_Containers()
  338. {
  339. var target = CreateTarget(itemCount: 6);
  340. target.ApplyTemplate();
  341. target.Measure(new Size(100, 100));
  342. target.Arrange(new Rect(0, 0, 100, 100));
  343. var expected = Enumerable.Range(0, 6).Select(x => $"Item {x}").ToList();
  344. var items = (ObservableCollection<string>)target.Items;
  345. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  346. Assert.Equal(expected, actual);
  347. expected = Enumerable.Range(0, 5).Select(x => $"Item {x}").ToList();
  348. target.Items = null;
  349. target.Items = expected;
  350. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  351. Assert.Equal(expected, actual);
  352. }
  353. [Fact]
  354. public void Scrolling_To_Partial_Last_Item_Then_Adding_Item_Updates_Containers()
  355. {
  356. var target = CreateTarget(itemCount: 10);
  357. var items = (IList<string>)target.Items;
  358. target.ApplyTemplate();
  359. target.Measure(new Size(100, 95));
  360. target.Arrange(new Rect(0, 0, 100, 95));
  361. ((ILogicalScrollable)target).Offset = new Vector(0, 1);
  362. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  363. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  364. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  365. Assert.Equal(expected, actual);
  366. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  367. items.Add("Item 10");
  368. expected = Enumerable.Range(1, 10).Select(x => $"Item {x}").ToList();
  369. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  370. Assert.Equal(expected, actual);
  371. Assert.Equal(0, ((IVirtualizingPanel)target.Panel).PixelOffset);
  372. }
  373. public class Vertical
  374. {
  375. [Fact]
  376. public void GetControlInDirection_Down_Should_Return_Existing_Container_If_Materialized()
  377. {
  378. var target = CreateTarget();
  379. target.ApplyTemplate();
  380. target.Measure(new Size(100, 100));
  381. target.Arrange(new Rect(0, 0, 100, 100));
  382. var from = target.Panel.Children[5];
  383. var result = ((ILogicalScrollable)target).GetControlInDirection(
  384. NavigationDirection.Down,
  385. from);
  386. Assert.Same(target.Panel.Children[6], result);
  387. }
  388. [Fact]
  389. public void GetControlInDirection_Down_Should_Scroll_If_Necessary()
  390. {
  391. var target = CreateTarget();
  392. target.ApplyTemplate();
  393. target.Measure(new Size(100, 100));
  394. target.Arrange(new Rect(0, 0, 100, 100));
  395. var from = target.Panel.Children[9];
  396. var result = ((ILogicalScrollable)target).GetControlInDirection(
  397. NavigationDirection.Down,
  398. from);
  399. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  400. Assert.Same(target.Panel.Children[9], result);
  401. }
  402. [Fact]
  403. public void GetControlInDirection_Down_Should_Scroll_If_Partially_Visible()
  404. {
  405. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  406. {
  407. var target = CreateTarget();
  408. var scroller = (ScrollContentPresenter)target.Parent;
  409. scroller.Measure(new Size(100, 95));
  410. scroller.Arrange(new Rect(0, 0, 100, 95));
  411. var from = target.Panel.Children[8];
  412. var result = ((ILogicalScrollable)target).GetControlInDirection(
  413. NavigationDirection.Down,
  414. from);
  415. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  416. Assert.Same(target.Panel.Children[8], result);
  417. }
  418. }
  419. [Fact]
  420. public void GetControlInDirection_Up_Should_Scroll_If_Partially_Visible_Item_Is_Currently_Shown()
  421. {
  422. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  423. {
  424. var target = CreateTarget();
  425. var scroller = (ScrollContentPresenter)target.Parent;
  426. scroller.Measure(new Size(100, 95));
  427. scroller.Arrange(new Rect(0, 0, 100, 95));
  428. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  429. var from = target.Panel.Children[1];
  430. var result = ((ILogicalScrollable)target).GetControlInDirection(
  431. NavigationDirection.Up,
  432. from);
  433. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  434. Assert.Same(target.Panel.Children[0], result);
  435. }
  436. }
  437. }
  438. public class Horizontal
  439. {
  440. [Fact]
  441. public void GetControlInDirection_Right_Should_Return_Existing_Container_If_Materialized()
  442. {
  443. var target = CreateTarget(orientation: Orientation.Horizontal);
  444. target.ApplyTemplate();
  445. target.Measure(new Size(100, 100));
  446. target.Arrange(new Rect(0, 0, 100, 100));
  447. var from = target.Panel.Children[5];
  448. var result = ((ILogicalScrollable)target).GetControlInDirection(
  449. NavigationDirection.Right,
  450. from);
  451. Assert.Same(target.Panel.Children[6], result);
  452. }
  453. [Fact]
  454. public void GetControlInDirection_Right_Should_Scroll_If_Necessary()
  455. {
  456. var target = CreateTarget(orientation: Orientation.Horizontal);
  457. target.ApplyTemplate();
  458. target.Measure(new Size(100, 100));
  459. target.Arrange(new Rect(0, 0, 100, 100));
  460. var from = target.Panel.Children[9];
  461. var result = ((ILogicalScrollable)target).GetControlInDirection(
  462. NavigationDirection.Right,
  463. from);
  464. Assert.Equal(new Vector(1, 0), ((ILogicalScrollable)target).Offset);
  465. Assert.Same(target.Panel.Children[9], result);
  466. }
  467. [Fact]
  468. public void GetControlInDirection_Right_Should_Scroll_If_Partially_Visible()
  469. {
  470. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  471. {
  472. var target = CreateTarget(orientation: Orientation.Horizontal);
  473. var scroller = (ScrollContentPresenter)target.Parent;
  474. scroller.Measure(new Size(95, 100));
  475. scroller.Arrange(new Rect(0, 0, 95, 100));
  476. var from = target.Panel.Children[8];
  477. var result = ((ILogicalScrollable)target).GetControlInDirection(
  478. NavigationDirection.Right,
  479. from);
  480. Assert.Equal(new Vector(1, 0), ((ILogicalScrollable)target).Offset);
  481. Assert.Same(target.Panel.Children[8], result);
  482. }
  483. }
  484. [Fact]
  485. public void GetControlInDirection_Left_Should_Scroll_If_Partially_Visible_Item_Is_Currently_Shown()
  486. {
  487. var target = CreateTarget(orientation: Orientation.Horizontal);
  488. target.ApplyTemplate();
  489. target.Measure(new Size(95, 100));
  490. target.Arrange(new Rect(0, 0, 95, 100));
  491. ((ILogicalScrollable)target).Offset = new Vector(11, 0);
  492. var from = target.Panel.Children[1];
  493. var result = ((ILogicalScrollable)target).GetControlInDirection(
  494. NavigationDirection.Left,
  495. from);
  496. Assert.Equal(new Vector(10, 0), ((ILogicalScrollable)target).Offset);
  497. Assert.Same(target.Panel.Children[0], result);
  498. }
  499. }
  500. public class WithContainers
  501. {
  502. [Fact]
  503. public void Scrolling_Less_Than_A_Page_Should_Move_Recycled_Items()
  504. {
  505. var target = CreateTarget();
  506. var items = (IList<string>)target.Items;
  507. target.ApplyTemplate();
  508. target.Measure(new Size(100, 100));
  509. target.Arrange(new Rect(0, 0, 100, 100));
  510. var containers = target.Panel.Children.ToList();
  511. var scroller = (ScrollContentPresenter)target.Parent;
  512. scroller.Offset = new Vector(0, 5);
  513. var scrolledContainers = containers
  514. .Skip(5)
  515. .Take(5)
  516. .Concat(containers.Take(5)).ToList();
  517. Assert.Equal(new Vector(0, 5), ((ILogicalScrollable)target).Offset);
  518. Assert.Equal(scrolledContainers, target.Panel.Children);
  519. for (var i = 0; i < target.Panel.Children.Count; ++i)
  520. {
  521. Assert.Equal(items[i + 5], target.Panel.Children[i].DataContext);
  522. }
  523. scroller.Offset = new Vector(0, 0);
  524. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  525. Assert.Equal(containers, target.Panel.Children);
  526. var dcs = target.Panel.Children.Select(x => x.DataContext).ToList();
  527. for (var i = 0; i < target.Panel.Children.Count; ++i)
  528. {
  529. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  530. }
  531. }
  532. [Fact]
  533. public void Scrolling_More_Than_A_Page_Should_Recycle_Items()
  534. {
  535. var target = CreateTarget(itemCount: 50);
  536. var items = (IList<string>)target.Items;
  537. target.ApplyTemplate();
  538. target.Measure(new Size(100, 100));
  539. target.Arrange(new Rect(0, 0, 100, 100));
  540. var containers = target.Panel.Children.ToList();
  541. var scroller = (ScrollContentPresenter)target.Parent;
  542. scroller.Offset = new Vector(0, 20);
  543. Assert.Equal(new Vector(0, 20), ((ILogicalScrollable)target).Offset);
  544. Assert.Equal(containers, target.Panel.Children);
  545. for (var i = 0; i < target.Panel.Children.Count; ++i)
  546. {
  547. Assert.Equal(items[i + 20], target.Panel.Children[i].DataContext);
  548. }
  549. scroller.Offset = new Vector(0, 0);
  550. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  551. Assert.Equal(containers, target.Panel.Children);
  552. for (var i = 0; i < target.Panel.Children.Count; ++i)
  553. {
  554. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  555. }
  556. }
  557. }
  558. private static ItemsPresenter CreateTarget(
  559. Orientation orientation = Orientation.Vertical,
  560. bool useContainers = true,
  561. int itemCount = 20,
  562. bool useAvaloniaList = false)
  563. {
  564. ItemsPresenter result;
  565. var itemsSource = Enumerable.Range(0, itemCount).Select(x => $"Item {x}");
  566. var items = useAvaloniaList ?
  567. (IEnumerable)new AvaloniaList<string>(itemsSource) :
  568. (IEnumerable)new ObservableCollection<string>(itemsSource);
  569. var scroller = new TestScroller
  570. {
  571. Content = result = new TestItemsPresenter(useContainers)
  572. {
  573. Items = items,
  574. ItemsPanel = VirtualizingPanelTemplate(orientation),
  575. ItemTemplate = ItemTemplate(),
  576. VirtualizationMode = ItemVirtualizationMode.Simple,
  577. }
  578. };
  579. scroller.UpdateChild();
  580. return result;
  581. }
  582. private static IDataTemplate ItemTemplate()
  583. {
  584. return new FuncDataTemplate<string>(x => new Canvas
  585. {
  586. Width = 10,
  587. Height = 10,
  588. });
  589. }
  590. private static ITemplate<IPanel> VirtualizingPanelTemplate(
  591. Orientation orientation = Orientation.Vertical)
  592. {
  593. return new FuncTemplate<IPanel>(() => new VirtualizingStackPanel
  594. {
  595. Orientation = orientation,
  596. });
  597. }
  598. private class TestScroller : ScrollContentPresenter, IRenderRoot
  599. {
  600. public IRenderQueueManager RenderQueueManager { get; }
  601. public Point PointToClient(Point point)
  602. {
  603. throw new NotImplementedException();
  604. }
  605. public Point PointToScreen(Point point)
  606. {
  607. throw new NotImplementedException();
  608. }
  609. }
  610. private class TestItemsPresenter : ItemsPresenter
  611. {
  612. private bool _useContainers;
  613. public TestItemsPresenter(bool useContainers)
  614. {
  615. _useContainers = useContainers;
  616. }
  617. protected override IItemContainerGenerator CreateItemContainerGenerator()
  618. {
  619. return _useContainers ?
  620. new ItemContainerGenerator<TestContainer>(this, TestContainer.ContentProperty, null) :
  621. new ItemContainerGenerator(this);
  622. }
  623. }
  624. private class TestContainer : ContentControl
  625. {
  626. public TestContainer()
  627. {
  628. Width = 10;
  629. Height = 10;
  630. }
  631. }
  632. }
  633. }