ItemsPresenterTests_Virtualization_Simple.cs 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848
  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. [Fact]
  374. public void Scrolling_To_Item_In_Zero_Sized_Presenter_Doesnt_Throw()
  375. {
  376. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  377. {
  378. var target = CreateTarget(itemCount: 10);
  379. var items = (IList<string>)target.Items;
  380. target.ApplyTemplate();
  381. target.Measure(Size.Empty);
  382. target.Arrange(Rect.Empty);
  383. // Check for issue #591: this should not throw.
  384. target.ScrollIntoView(items[0]);
  385. }
  386. }
  387. public class Vertical
  388. {
  389. [Fact]
  390. public void GetControlInDirection_Down_Should_Return_Existing_Container_If_Materialized()
  391. {
  392. var target = CreateTarget();
  393. target.ApplyTemplate();
  394. target.Measure(new Size(100, 100));
  395. target.Arrange(new Rect(0, 0, 100, 100));
  396. var from = target.Panel.Children[5];
  397. var result = ((ILogicalScrollable)target).GetControlInDirection(
  398. NavigationDirection.Down,
  399. from);
  400. Assert.Same(target.Panel.Children[6], result);
  401. }
  402. [Fact]
  403. public void GetControlInDirection_Down_Should_Scroll_If_Necessary()
  404. {
  405. var target = CreateTarget();
  406. target.ApplyTemplate();
  407. target.Measure(new Size(100, 100));
  408. target.Arrange(new Rect(0, 0, 100, 100));
  409. var from = target.Panel.Children[9];
  410. var result = ((ILogicalScrollable)target).GetControlInDirection(
  411. NavigationDirection.Down,
  412. from);
  413. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  414. Assert.Same(target.Panel.Children[9], result);
  415. }
  416. [Fact]
  417. public void GetControlInDirection_Down_Should_Scroll_If_Partially_Visible()
  418. {
  419. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  420. {
  421. var target = CreateTarget();
  422. var scroller = (ScrollContentPresenter)target.Parent;
  423. scroller.Measure(new Size(100, 95));
  424. scroller.Arrange(new Rect(0, 0, 100, 95));
  425. var from = target.Panel.Children[8];
  426. var result = ((ILogicalScrollable)target).GetControlInDirection(
  427. NavigationDirection.Down,
  428. from);
  429. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  430. Assert.Same(target.Panel.Children[8], result);
  431. }
  432. }
  433. [Fact]
  434. public void GetControlInDirection_Up_Should_Scroll_If_Partially_Visible_Item_Is_Currently_Shown()
  435. {
  436. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  437. {
  438. var target = CreateTarget();
  439. var scroller = (ScrollContentPresenter)target.Parent;
  440. scroller.Measure(new Size(100, 95));
  441. scroller.Arrange(new Rect(0, 0, 100, 95));
  442. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  443. var from = target.Panel.Children[1];
  444. var result = ((ILogicalScrollable)target).GetControlInDirection(
  445. NavigationDirection.Up,
  446. from);
  447. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  448. Assert.Same(target.Panel.Children[0], result);
  449. }
  450. }
  451. }
  452. public class Horizontal
  453. {
  454. [Fact]
  455. public void GetControlInDirection_Right_Should_Return_Existing_Container_If_Materialized()
  456. {
  457. var target = CreateTarget(orientation: Orientation.Horizontal);
  458. target.ApplyTemplate();
  459. target.Measure(new Size(100, 100));
  460. target.Arrange(new Rect(0, 0, 100, 100));
  461. var from = target.Panel.Children[5];
  462. var result = ((ILogicalScrollable)target).GetControlInDirection(
  463. NavigationDirection.Right,
  464. from);
  465. Assert.Same(target.Panel.Children[6], result);
  466. }
  467. [Fact]
  468. public void GetControlInDirection_Right_Should_Scroll_If_Necessary()
  469. {
  470. var target = CreateTarget(orientation: Orientation.Horizontal);
  471. target.ApplyTemplate();
  472. target.Measure(new Size(100, 100));
  473. target.Arrange(new Rect(0, 0, 100, 100));
  474. var from = target.Panel.Children[9];
  475. var result = ((ILogicalScrollable)target).GetControlInDirection(
  476. NavigationDirection.Right,
  477. from);
  478. Assert.Equal(new Vector(1, 0), ((ILogicalScrollable)target).Offset);
  479. Assert.Same(target.Panel.Children[9], result);
  480. }
  481. [Fact]
  482. public void GetControlInDirection_Right_Should_Scroll_If_Partially_Visible()
  483. {
  484. using (UnitTestApplication.Start(TestServices.RealLayoutManager))
  485. {
  486. var target = CreateTarget(orientation: Orientation.Horizontal);
  487. var scroller = (ScrollContentPresenter)target.Parent;
  488. scroller.Measure(new Size(95, 100));
  489. scroller.Arrange(new Rect(0, 0, 95, 100));
  490. var from = target.Panel.Children[8];
  491. var result = ((ILogicalScrollable)target).GetControlInDirection(
  492. NavigationDirection.Right,
  493. from);
  494. Assert.Equal(new Vector(1, 0), ((ILogicalScrollable)target).Offset);
  495. Assert.Same(target.Panel.Children[8], result);
  496. }
  497. }
  498. [Fact]
  499. public void GetControlInDirection_Left_Should_Scroll_If_Partially_Visible_Item_Is_Currently_Shown()
  500. {
  501. var target = CreateTarget(orientation: Orientation.Horizontal);
  502. target.ApplyTemplate();
  503. target.Measure(new Size(95, 100));
  504. target.Arrange(new Rect(0, 0, 95, 100));
  505. ((ILogicalScrollable)target).Offset = new Vector(11, 0);
  506. var from = target.Panel.Children[1];
  507. var result = ((ILogicalScrollable)target).GetControlInDirection(
  508. NavigationDirection.Left,
  509. from);
  510. Assert.Equal(new Vector(10, 0), ((ILogicalScrollable)target).Offset);
  511. Assert.Same(target.Panel.Children[0], result);
  512. }
  513. }
  514. public class WithContainers
  515. {
  516. [Fact]
  517. public void Scrolling_Less_Than_A_Page_Should_Move_Recycled_Items()
  518. {
  519. var target = CreateTarget();
  520. var items = (IList<string>)target.Items;
  521. target.ApplyTemplate();
  522. target.Measure(new Size(100, 100));
  523. target.Arrange(new Rect(0, 0, 100, 100));
  524. var containers = target.Panel.Children.ToList();
  525. var scroller = (ScrollContentPresenter)target.Parent;
  526. scroller.Offset = new Vector(0, 5);
  527. var scrolledContainers = containers
  528. .Skip(5)
  529. .Take(5)
  530. .Concat(containers.Take(5)).ToList();
  531. Assert.Equal(new Vector(0, 5), ((ILogicalScrollable)target).Offset);
  532. Assert.Equal(scrolledContainers, target.Panel.Children);
  533. for (var i = 0; i < target.Panel.Children.Count; ++i)
  534. {
  535. Assert.Equal(items[i + 5], target.Panel.Children[i].DataContext);
  536. }
  537. scroller.Offset = new Vector(0, 0);
  538. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  539. Assert.Equal(containers, target.Panel.Children);
  540. var dcs = target.Panel.Children.Select(x => x.DataContext).ToList();
  541. for (var i = 0; i < target.Panel.Children.Count; ++i)
  542. {
  543. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  544. }
  545. }
  546. [Fact]
  547. public void Scrolling_More_Than_A_Page_Should_Recycle_Items()
  548. {
  549. var target = CreateTarget(itemCount: 50);
  550. var items = (IList<string>)target.Items;
  551. target.ApplyTemplate();
  552. target.Measure(new Size(100, 100));
  553. target.Arrange(new Rect(0, 0, 100, 100));
  554. var containers = target.Panel.Children.ToList();
  555. var scroller = (ScrollContentPresenter)target.Parent;
  556. scroller.Offset = new Vector(0, 20);
  557. Assert.Equal(new Vector(0, 20), ((ILogicalScrollable)target).Offset);
  558. Assert.Equal(containers, target.Panel.Children);
  559. for (var i = 0; i < target.Panel.Children.Count; ++i)
  560. {
  561. Assert.Equal(items[i + 20], target.Panel.Children[i].DataContext);
  562. }
  563. scroller.Offset = new Vector(0, 0);
  564. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  565. Assert.Equal(containers, target.Panel.Children);
  566. for (var i = 0; i < target.Panel.Children.Count; ++i)
  567. {
  568. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  569. }
  570. }
  571. }
  572. private static ItemsPresenter CreateTarget(
  573. Orientation orientation = Orientation.Vertical,
  574. bool useContainers = true,
  575. int itemCount = 20,
  576. bool useAvaloniaList = false)
  577. {
  578. ItemsPresenter result;
  579. var itemsSource = Enumerable.Range(0, itemCount).Select(x => $"Item {x}");
  580. var items = useAvaloniaList ?
  581. (IEnumerable)new AvaloniaList<string>(itemsSource) :
  582. (IEnumerable)new ObservableCollection<string>(itemsSource);
  583. var scroller = new TestScroller
  584. {
  585. Content = result = new TestItemsPresenter(useContainers)
  586. {
  587. Items = items,
  588. ItemsPanel = VirtualizingPanelTemplate(orientation),
  589. ItemTemplate = ItemTemplate(),
  590. VirtualizationMode = ItemVirtualizationMode.Simple,
  591. }
  592. };
  593. scroller.UpdateChild();
  594. return result;
  595. }
  596. private static IDataTemplate ItemTemplate()
  597. {
  598. return new FuncDataTemplate<string>(x => new Canvas
  599. {
  600. Width = 10,
  601. Height = 10,
  602. });
  603. }
  604. private static ITemplate<IPanel> VirtualizingPanelTemplate(
  605. Orientation orientation = Orientation.Vertical)
  606. {
  607. return new FuncTemplate<IPanel>(() => new VirtualizingStackPanel
  608. {
  609. Orientation = orientation,
  610. });
  611. }
  612. private class TestScroller : ScrollContentPresenter, IRenderRoot
  613. {
  614. public IRenderQueueManager RenderQueueManager { get; }
  615. public Point PointToClient(Point point)
  616. {
  617. throw new NotImplementedException();
  618. }
  619. public Point PointToScreen(Point point)
  620. {
  621. throw new NotImplementedException();
  622. }
  623. }
  624. private class TestItemsPresenter : ItemsPresenter
  625. {
  626. private bool _useContainers;
  627. public TestItemsPresenter(bool useContainers)
  628. {
  629. _useContainers = useContainers;
  630. }
  631. protected override IItemContainerGenerator CreateItemContainerGenerator()
  632. {
  633. return _useContainers ?
  634. new ItemContainerGenerator<TestContainer>(this, TestContainer.ContentProperty, null) :
  635. new ItemContainerGenerator(this);
  636. }
  637. }
  638. private class TestContainer : ContentControl
  639. {
  640. public TestContainer()
  641. {
  642. Width = 10;
  643. Height = 10;
  644. }
  645. }
  646. }
  647. }