ItemsPresenterTests_Virtualization_Simple.cs 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107
  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.Layout;
  15. using Avalonia.Platform;
  16. using Avalonia.Rendering;
  17. using Avalonia.Styling;
  18. using Avalonia.UnitTests;
  19. using Xunit;
  20. namespace Avalonia.Controls.UnitTests.Presenters
  21. {
  22. public class ItemsPresenterTests_Virtualization_Simple
  23. {
  24. [Fact]
  25. public void Should_Return_Items_Count_For_Extent_Vertical()
  26. {
  27. var target = CreateTarget();
  28. target.ApplyTemplate();
  29. Assert.Equal(new Size(0, 20), ((ILogicalScrollable)target).Extent);
  30. }
  31. [Fact]
  32. public void Should_Return_Items_Count_For_Extent_Horizontal()
  33. {
  34. var target = CreateTarget(orientation: Orientation.Horizontal);
  35. target.ApplyTemplate();
  36. Assert.Equal(new Size(20, 0), ((ILogicalScrollable)target).Extent);
  37. }
  38. [Fact]
  39. public void Should_Have_Number_Of_Visible_Items_As_Viewport_Vertical()
  40. {
  41. var target = CreateTarget();
  42. target.ApplyTemplate();
  43. target.Measure(new Size(100, 100));
  44. target.Arrange(new Rect(0, 0, 100, 100));
  45. Assert.Equal(new Size(100, 10), ((ILogicalScrollable)target).Viewport);
  46. }
  47. [Fact]
  48. public void Should_Have_Number_Of_Visible_Items_As_Viewport_Horizontal()
  49. {
  50. var target = CreateTarget(orientation: Orientation.Horizontal);
  51. target.ApplyTemplate();
  52. target.Measure(new Size(100, 100));
  53. target.Arrange(new Rect(0, 0, 100, 100));
  54. Assert.Equal(new Size(10, 100), ((ILogicalScrollable)target).Viewport);
  55. }
  56. [Fact]
  57. public void Should_Add_Containers_When_Panel_Is_Not_Full()
  58. {
  59. var target = CreateTarget(itemCount: 5);
  60. var items = (IList<string>)target.Items;
  61. target.ApplyTemplate();
  62. target.Measure(new Size(100, 100));
  63. target.Arrange(new Rect(0, 0, 100, 100));
  64. Assert.Equal(5, target.Panel.Children.Count);
  65. items.Add("New Item");
  66. Assert.Equal(6, target.Panel.Children.Count);
  67. }
  68. [Fact]
  69. public void Should_Remove_Items_When_Control_Is_Shrank()
  70. {
  71. var target = CreateTarget();
  72. var items = (IList<string>)target.Items;
  73. target.ApplyTemplate();
  74. target.Measure(new Size(100, 100));
  75. target.Arrange(new Rect(0, 0, 100, 100));
  76. Assert.Equal(10, target.Panel.Children.Count);
  77. target.Measure(new Size(100, 80));
  78. target.Arrange(new Rect(0, 0, 100, 80));
  79. Assert.Equal(8, target.Panel.Children.Count);
  80. }
  81. [Fact]
  82. public void Should_Add_New_Containers_At_Top_When_Control_Is_Scrolled_To_Bottom_And_Enlarged()
  83. {
  84. var target = CreateTarget();
  85. var items = (IList<string>)target.Items;
  86. target.ApplyTemplate();
  87. target.Measure(new Size(100, 100));
  88. target.Arrange(new Rect(0, 0, 100, 100));
  89. Assert.Equal(10, target.Panel.Children.Count);
  90. ((IScrollable)target).Offset = new Vector(0, 10);
  91. target.Measure(new Size(120, 120));
  92. target.Arrange(new Rect(0, 0, 100, 120));
  93. Assert.Equal(12, target.Panel.Children.Count);
  94. for (var i = 0; i < target.Panel.Children.Count; ++i)
  95. {
  96. Assert.Equal(items[i + 8], target.Panel.Children[i].DataContext);
  97. }
  98. }
  99. [Fact]
  100. public void Should_Update_Containers_When_Items_Changes()
  101. {
  102. var target = CreateTarget();
  103. target.ApplyTemplate();
  104. target.Measure(new Size(100, 100));
  105. target.Arrange(new Rect(0, 0, 100, 100));
  106. target.Items = new[] { "foo", "bar", "baz" };
  107. Assert.Equal(3, target.Panel.Children.Count);
  108. }
  109. [Fact]
  110. public void Should_Decrease_The_Viewport_Size_By_One_If_There_Is_A_Partial_Item()
  111. {
  112. var target = CreateTarget();
  113. target.ApplyTemplate();
  114. target.Measure(new Size(100, 95));
  115. target.Arrange(new Rect(0, 0, 100, 95));
  116. Assert.Equal(new Size(100, 9), ((ILogicalScrollable)target).Viewport);
  117. }
  118. [Fact]
  119. public void Moving_To_And_From_The_End_With_Partial_Item_Should_Set_Panel_PixelOffset()
  120. {
  121. var target = CreateTarget();
  122. target.ApplyTemplate();
  123. target.Measure(new Size(100, 95));
  124. target.Arrange(new Rect(0, 0, 100, 95));
  125. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  126. var minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  127. Assert.Equal(new Vector(0, 11), ((ILogicalScrollable)target).Offset);
  128. Assert.Equal(10, minIndex);
  129. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  130. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  131. minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  132. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  133. Assert.Equal(10, minIndex);
  134. Assert.Equal(0, ((IVirtualizingPanel)target.Panel).PixelOffset);
  135. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  136. minIndex = target.ItemContainerGenerator.Containers.Min(x => x.Index);
  137. Assert.Equal(new Vector(0, 11), ((ILogicalScrollable)target).Offset);
  138. Assert.Equal(10, minIndex);
  139. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  140. }
  141. [Fact]
  142. public void Inserting_Items_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. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  149. var expected = Enumerable.Range(5, 10).Select(x => $"Item {x}").ToList();
  150. var items = (ObservableCollection<string>)target.Items;
  151. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  152. Assert.Equal(expected, actual);
  153. items.Insert(6, "Inserted");
  154. expected.Insert(1, "Inserted");
  155. expected.RemoveAt(expected.Count - 1);
  156. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  157. Assert.Equal(expected, actual);
  158. }
  159. [Fact]
  160. public void Inserting_Items_Before_Visibile_Containers_Should_Update_Containers()
  161. {
  162. var target = CreateTarget();
  163. target.ApplyTemplate();
  164. target.Measure(new Size(100, 100));
  165. target.Arrange(new Rect(0, 0, 100, 100));
  166. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  167. var expected = Enumerable.Range(5, 10).Select(x => $"Item {x}").ToList();
  168. var items = (ObservableCollection<string>)target.Items;
  169. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  170. Assert.Equal(expected, actual);
  171. items.Insert(0, "Inserted");
  172. expected = Enumerable.Range(4, 10).Select(x => $"Item {x}").ToList();
  173. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  174. Assert.Equal(expected, actual);
  175. }
  176. [Fact]
  177. public void Removing_First_Materialized_Item_Should_Update_Containers()
  178. {
  179. var target = CreateTarget();
  180. target.ApplyTemplate();
  181. target.Measure(new Size(100, 100));
  182. target.Arrange(new Rect(0, 0, 100, 100));
  183. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  184. var items = (ObservableCollection<string>)target.Items;
  185. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  186. Assert.Equal(expected, actual);
  187. items.RemoveAt(0);
  188. expected = Enumerable.Range(1, 10).Select(x => $"Item {x}").ToList();
  189. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  190. Assert.Equal(expected, actual);
  191. }
  192. [Fact]
  193. public void Removing_Items_From_Middle_Should_Update_Containers_When_All_Items_Visible()
  194. {
  195. var target = CreateTarget();
  196. target.ApplyTemplate();
  197. target.Measure(new Size(100, 200));
  198. target.Arrange(new Rect(0, 0, 100, 200));
  199. var items = (ObservableCollection<string>)target.Items;
  200. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  201. Assert.Equal(items, actual);
  202. items.RemoveAt(2);
  203. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  204. Assert.Equal(items, actual);
  205. items.RemoveAt(items.Count - 2);
  206. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  207. Assert.Equal(items, actual);
  208. }
  209. [Fact]
  210. public void Removing_Last_Item_Should_Update_Containers_When_All_Items_Visible()
  211. {
  212. var target = CreateTarget();
  213. target.ApplyTemplate();
  214. target.Measure(new Size(100, 200));
  215. target.Arrange(new Rect(0, 0, 100, 200));
  216. ((ILogicalScrollable)target).Offset = new Vector(0, 5);
  217. var expected = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList();
  218. var items = (ObservableCollection<string>)target.Items;
  219. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  220. Assert.Equal(expected, actual);
  221. items.Remove(items.Last());
  222. expected.Remove(expected.Last());
  223. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  224. Assert.Equal(expected, actual);
  225. }
  226. [Fact]
  227. public void Removing_Items_When_Scrolled_To_End_Should_Recyle_Containers_At_Top()
  228. {
  229. var target = CreateTarget(useAvaloniaList: true);
  230. target.ApplyTemplate();
  231. target.Measure(new Size(100, 100));
  232. target.Arrange(new Rect(0, 0, 100, 100));
  233. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  234. var expected = Enumerable.Range(10, 10).Select(x => $"Item {x}").ToList();
  235. var items = (AvaloniaList<string>)target.Items;
  236. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  237. Assert.Equal(expected, actual);
  238. items.RemoveRange(18, 2);
  239. expected = Enumerable.Range(8, 10).Select(x => $"Item {x}").ToList();
  240. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  241. Assert.Equal(expected, actual);
  242. }
  243. [Fact]
  244. public void Removing_Items_When_Scrolled_To_Near_End_Should_Recycle_Containers_At_Bottom_And_Top()
  245. {
  246. var target = CreateTarget(useAvaloniaList: true);
  247. target.ApplyTemplate();
  248. target.Measure(new Size(100, 100));
  249. target.Arrange(new Rect(0, 0, 100, 100));
  250. ((ILogicalScrollable)target).Offset = new Vector(0, 9);
  251. var expected = Enumerable.Range(9, 10).Select(x => $"Item {x}").ToList();
  252. var items = (AvaloniaList<string>)target.Items;
  253. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  254. Assert.Equal(expected, actual);
  255. items.RemoveRange(15, 3);
  256. expected = Enumerable.Range(7, 8).Select(x => $"Item {x}")
  257. .Concat(Enumerable.Range(18, 2).Select(x => $"Item {x}"))
  258. .ToList();
  259. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  260. Assert.Equal(expected, actual);
  261. }
  262. [Fact]
  263. public void Measuring_To_Infinity_When_Scrolled_To_End_Should_Not_Throw()
  264. {
  265. var target = CreateTarget(useAvaloniaList: true);
  266. target.ApplyTemplate();
  267. target.Measure(new Size(100, 100));
  268. target.Arrange(new Rect(0, 0, 100, 100));
  269. ((ILogicalScrollable)target).Offset = new Vector(0, 10);
  270. // Check for issue #589: this should not throw.
  271. target.Measure(Size.Infinity);
  272. var expected = Enumerable.Range(0, 20).Select(x => $"Item {x}").ToList();
  273. var items = (AvaloniaList<string>)target.Items;
  274. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  275. Assert.Equal(expected, actual);
  276. }
  277. [Fact]
  278. public void Replacing_Items_Should_Update_Containers()
  279. {
  280. var target = CreateTarget();
  281. target.ApplyTemplate();
  282. target.Measure(new Size(100, 100));
  283. target.Arrange(new Rect(0, 0, 100, 100));
  284. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  285. var items = (ObservableCollection<string>)target.Items;
  286. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  287. Assert.Equal(expected, actual);
  288. items[4] = expected[4] = "Replaced";
  289. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  290. Assert.Equal(expected, actual);
  291. }
  292. [Fact]
  293. public void Moving_Items_Should_Update_Containers()
  294. {
  295. var target = CreateTarget();
  296. target.ApplyTemplate();
  297. target.Measure(new Size(100, 100));
  298. target.Arrange(new Rect(0, 0, 100, 100));
  299. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  300. var items = (ObservableCollection<string>)target.Items;
  301. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  302. Assert.Equal(expected, actual);
  303. items.Move(4, 8);
  304. var i = expected[4];
  305. expected.RemoveAt(4);
  306. expected.Insert(8, i);
  307. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  308. Assert.Equal(expected, actual);
  309. }
  310. [Fact]
  311. public void Setting_Items_To_Null_Should_Remove_Containers()
  312. {
  313. var target = CreateTarget();
  314. target.ApplyTemplate();
  315. target.Measure(new Size(100, 100));
  316. target.Arrange(new Rect(0, 0, 100, 100));
  317. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  318. var items = (ObservableCollection<string>)target.Items;
  319. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  320. Assert.Equal(expected, actual);
  321. target.Items = null;
  322. Assert.Empty(target.Panel.Children);
  323. }
  324. [Fact]
  325. public void Reassigning_Items_Should_Create_Containers()
  326. {
  327. var target = CreateTarget(itemCount: 5);
  328. target.ApplyTemplate();
  329. target.Measure(new Size(100, 100));
  330. target.Arrange(new Rect(0, 0, 100, 100));
  331. var expected = Enumerable.Range(0, 5).Select(x => $"Item {x}").ToList();
  332. var items = (ObservableCollection<string>)target.Items;
  333. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  334. Assert.Equal(expected, actual);
  335. expected = Enumerable.Range(0, 6).Select(x => $"Item {x}").ToList();
  336. target.Items = expected;
  337. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  338. Assert.Equal(expected, actual);
  339. }
  340. [Fact]
  341. public void Inserting_Then_Removing_Should_Add_Remove_Containers()
  342. {
  343. var items = new AvaloniaList<string>(Enumerable.Range(0, 5).Select(x => $"Item {x}"));
  344. var toAdd = Enumerable.Range(0, 3).Select(x => $"Added Item {x}").ToArray();
  345. var target = new ItemsPresenter
  346. {
  347. VirtualizationMode = ItemVirtualizationMode.None,
  348. Items = items,
  349. ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Height = 10 }),
  350. };
  351. target.ApplyTemplate();
  352. Assert.Equal(items.Count, target.Panel.Children.Count);
  353. int addIndex = 1;
  354. foreach (var item in toAdd)
  355. {
  356. items.Insert(addIndex++, item);
  357. }
  358. Assert.Equal(items.Count, target.Panel.Children.Count);
  359. foreach (var item in toAdd)
  360. {
  361. items.Remove(item);
  362. }
  363. Assert.Equal(items.Count, target.Panel.Children.Count);
  364. }
  365. [Fact]
  366. public void Reassigning_Items_Should_Remove_Containers()
  367. {
  368. var target = CreateTarget(itemCount: 6);
  369. target.ApplyTemplate();
  370. target.Measure(new Size(100, 100));
  371. target.Arrange(new Rect(0, 0, 100, 100));
  372. var expected = Enumerable.Range(0, 6).Select(x => $"Item {x}").ToList();
  373. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  374. Assert.Equal(expected, actual);
  375. expected = Enumerable.Range(0, 5).Select(x => $"Item {x}").ToList();
  376. target.Items = expected;
  377. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  378. Assert.Equal(expected, actual);
  379. }
  380. [Fact]
  381. public void Clearing_Items_And_ReAdding_Should_Remove_Containers()
  382. {
  383. var target = CreateTarget(itemCount: 6);
  384. target.ApplyTemplate();
  385. target.Measure(new Size(100, 100));
  386. target.Arrange(new Rect(0, 0, 100, 100));
  387. var expected = Enumerable.Range(0, 6).Select(x => $"Item {x}").ToList();
  388. var items = (ObservableCollection<string>)target.Items;
  389. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  390. Assert.Equal(expected, actual);
  391. expected = Enumerable.Range(0, 5).Select(x => $"Item {x}").ToList();
  392. target.Items = null;
  393. target.Items = expected;
  394. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  395. Assert.Equal(expected, actual);
  396. }
  397. [Fact]
  398. public void Scrolling_To_Partial_Last_Item_Then_Adding_Item_Updates_Containers()
  399. {
  400. var target = CreateTarget(itemCount: 10);
  401. var items = (IList<string>)target.Items;
  402. target.ApplyTemplate();
  403. target.Measure(new Size(100, 95));
  404. target.Arrange(new Rect(0, 0, 100, 95));
  405. ((ILogicalScrollable)target).Offset = new Vector(0, 1);
  406. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  407. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  408. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  409. Assert.Equal(expected, actual);
  410. Assert.Equal(10, ((IVirtualizingPanel)target.Panel).PixelOffset);
  411. items.Add("Item 10");
  412. expected = Enumerable.Range(1, 10).Select(x => $"Item {x}").ToList();
  413. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  414. Assert.Equal(expected, actual);
  415. Assert.Equal(0, ((IVirtualizingPanel)target.Panel).PixelOffset);
  416. }
  417. [Fact]
  418. public void Scrolling_To_Item_In_Zero_Sized_Presenter_Doesnt_Throw()
  419. {
  420. using (UnitTestApplication.Start(new TestServices()))
  421. {
  422. var target = CreateTarget(itemCount: 10);
  423. var items = (IList<string>)target.Items;
  424. target.ApplyTemplate();
  425. target.Measure(Size.Empty);
  426. target.Arrange(Rect.Empty);
  427. // Check for issue #591: this should not throw.
  428. target.ScrollIntoView(items[0]);
  429. }
  430. }
  431. [Fact]
  432. public void InsertRange_Items_Should_Update_Containers()
  433. {
  434. var target = CreateTarget(useAvaloniaList: true);
  435. target.ApplyTemplate();
  436. target.Measure(new Size(100, 100));
  437. target.Arrange(new Rect(0, 0, 100, 100));
  438. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  439. var items = (AvaloniaList<string>)target.Items;
  440. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  441. Assert.Equal(expected, actual);
  442. var toAdd = Enumerable.Range(0, 3).Select(x => $"New Item {x}").ToList();
  443. int index = 1;
  444. items.InsertRange(index, toAdd);
  445. expected.InsertRange(index, toAdd);
  446. expected.RemoveRange(10, toAdd.Count);
  447. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  448. Assert.Equal(expected, actual);
  449. }
  450. [Fact]
  451. public void InsertRange_Items_Before_Last_Should_Update_Containers()
  452. {
  453. var target = CreateTarget(useAvaloniaList: true);
  454. target.ApplyTemplate();
  455. target.Measure(new Size(100, 100));
  456. target.Arrange(new Rect(0, 0, 100, 100));
  457. var expected = Enumerable.Range(0, 10).Select(x => $"Item {x}").ToList();
  458. var items = (AvaloniaList<string>)target.Items;
  459. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  460. Assert.Equal(expected, actual);
  461. var toAdd = Enumerable.Range(0, 3).Select(x => $"New Item {x}").ToList();
  462. int index = 8;
  463. items.InsertRange(index, toAdd);
  464. expected.InsertRange(index, toAdd);
  465. expected.RemoveRange(10, toAdd.Count);
  466. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  467. Assert.Equal(expected, actual);
  468. }
  469. [Fact]
  470. public void RemoveRange_Items_Should_Update_Containers()
  471. {
  472. var target = CreateTarget(useAvaloniaList: true);
  473. target.ApplyTemplate();
  474. target.Measure(new Size(100, 100));
  475. target.Arrange(new Rect(0, 0, 100, 100));
  476. var expected = Enumerable.Range(0, 13).Select(x => $"Item {x}").ToList();
  477. var items = (AvaloniaList<string>)target.Items;
  478. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  479. Assert.Equal(expected.Take(10), actual);
  480. int index = 5;
  481. int count = 3;
  482. items.RemoveRange(index, count);
  483. expected.RemoveRange(index, count);
  484. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  485. Assert.Equal(expected, actual);
  486. }
  487. [Fact]
  488. public void RemoveRange_Items_Before_Last_Should_Update_Containers()
  489. {
  490. var target = CreateTarget(useAvaloniaList: true);
  491. target.ApplyTemplate();
  492. target.Measure(new Size(100, 100));
  493. target.Arrange(new Rect(0, 0, 100, 100));
  494. var expected = Enumerable.Range(0, 13).Select(x => $"Item {x}").ToList();
  495. var items = (AvaloniaList<string>)target.Items;
  496. var actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  497. Assert.Equal(expected.Take(10), actual);
  498. int index = 8;
  499. int count = 3;
  500. items.RemoveRange(index, count);
  501. expected.RemoveRange(index, count);
  502. actual = target.Panel.Children.Select(x => x.DataContext).ToList();
  503. Assert.Equal(expected, actual);
  504. }
  505. [Fact]
  506. public void Should_Add_Containers_For_Items_After_Clear()
  507. {
  508. var target = CreateTarget(itemCount: 10);
  509. var defaultItems = (IList<string>)target.Items;
  510. var items = new AvaloniaList<string>(defaultItems);
  511. target.Items = items;
  512. target.ApplyTemplate();
  513. target.Measure(new Size(100, 100));
  514. target.Arrange(new Rect(target.DesiredSize));
  515. Assert.Equal(10, target.Panel.Children.Count);
  516. items.Clear();
  517. target.Panel.Measure(new Size(100, 100));
  518. target.Panel.Arrange(new Rect(target.Panel.DesiredSize));
  519. target.Measure(new Size(100, 100));
  520. target.Arrange(new Rect(target.DesiredSize));
  521. Assert.Empty(target.Panel.Children);
  522. items.AddRange(defaultItems.Select(s => s + " new"));
  523. target.Panel.Measure(new Size(100, 100));
  524. target.Panel.Arrange(new Rect(target.Panel.DesiredSize));
  525. target.Measure(new Size(100, 100));
  526. target.Arrange(new Rect(target.DesiredSize));
  527. Assert.Equal(10, target.Panel.Children.Count);
  528. }
  529. [Fact]
  530. public void Scroll_To_Last_Should_Work()
  531. {
  532. var target = CreateTarget(itemCount: 11);
  533. var scroller = (TestScroller)target.Parent;
  534. scroller.Width = scroller.Height = 100;
  535. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  536. var last = (target.Items as IList)[10];
  537. target.ScrollIntoView(last);
  538. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  539. Assert.Same(target.Panel.Children[9].DataContext, last);
  540. }
  541. [Fact]
  542. public void Second_Scroll_To_Last_Should_Work()
  543. {
  544. var target = CreateTarget(itemCount: 11);
  545. var scroller = (TestScroller)target.Parent;
  546. scroller.Width = scroller.Height = 100;
  547. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  548. var last = (target.Items as IList)[10];
  549. target.ScrollIntoView(last);
  550. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  551. Assert.Same(target.Panel.Children[9].DataContext, last);
  552. target.ScrollIntoView(last);
  553. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  554. Assert.Same(target.Panel.Children[9].DataContext, last);
  555. }
  556. [Fact]
  557. public void Scrolling_Less_Than_A_Page_Should_Move_Recycled_Items()
  558. {
  559. var target = CreateTarget();
  560. var items = (IList<string>)target.Items;
  561. target.ApplyTemplate();
  562. target.Measure(new Size(100, 100));
  563. target.Arrange(new Rect(0, 0, 100, 100));
  564. var containers = target.Panel.Children.ToList();
  565. var scroller = (ScrollContentPresenter)target.Parent;
  566. scroller.Offset = new Vector(0, 5);
  567. var scrolledContainers = containers
  568. .Skip(5)
  569. .Take(5)
  570. .Concat(containers.Take(5)).ToList();
  571. Assert.Equal(new Vector(0, 5), ((ILogicalScrollable)target).Offset);
  572. Assert.Equal(scrolledContainers, target.Panel.Children);
  573. for (var i = 0; i < target.Panel.Children.Count; ++i)
  574. {
  575. Assert.Equal(items[i + 5], target.Panel.Children[i].DataContext);
  576. }
  577. scroller.Offset = new Vector(0, 0);
  578. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  579. Assert.Equal(containers, target.Panel.Children);
  580. var dcs = target.Panel.Children.Select(x => x.DataContext).ToList();
  581. for (var i = 0; i < target.Panel.Children.Count; ++i)
  582. {
  583. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  584. }
  585. }
  586. [Fact]
  587. public void Scrolling_More_Than_A_Page_Should_Recycle_Items()
  588. {
  589. var target = CreateTarget(itemCount: 50);
  590. var items = (IList<string>)target.Items;
  591. target.ApplyTemplate();
  592. target.Measure(new Size(100, 100));
  593. target.Arrange(new Rect(0, 0, 100, 100));
  594. var containers = target.Panel.Children.ToList();
  595. var scroller = (ScrollContentPresenter)target.Parent;
  596. scroller.Offset = new Vector(0, 20);
  597. Assert.Equal(new Vector(0, 20), ((ILogicalScrollable)target).Offset);
  598. Assert.Equal(containers, target.Panel.Children);
  599. for (var i = 0; i < target.Panel.Children.Count; ++i)
  600. {
  601. Assert.Equal(items[i + 20], target.Panel.Children[i].DataContext);
  602. }
  603. scroller.Offset = new Vector(0, 0);
  604. Assert.Equal(new Vector(0, 0), ((ILogicalScrollable)target).Offset);
  605. Assert.Equal(containers, target.Panel.Children);
  606. for (var i = 0; i < target.Panel.Children.Count; ++i)
  607. {
  608. Assert.Equal(items[i], target.Panel.Children[i].DataContext);
  609. }
  610. }
  611. public class Vertical
  612. {
  613. [Fact]
  614. public void GetControlInDirection_Down_Should_Return_Existing_Container_If_Materialized()
  615. {
  616. var target = CreateTarget();
  617. var scroller = (TestScroller)target.Parent;
  618. scroller.Width = scroller.Height = 100;
  619. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  620. var from = target.Panel.Children[5];
  621. var result = ((ILogicalScrollable)target).GetControlInDirection(
  622. NavigationDirection.Down,
  623. from);
  624. Assert.Same(target.Panel.Children[6], result);
  625. }
  626. [Fact]
  627. public void GetControlInDirection_Down_Should_Scroll_If_Necessary()
  628. {
  629. var target = CreateTarget();
  630. var scroller = (TestScroller)target.Parent;
  631. scroller.Width = scroller.Height = 100;
  632. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  633. var from = target.Panel.Children[9];
  634. var result = ((ILogicalScrollable)target).GetControlInDirection(
  635. NavigationDirection.Down,
  636. from);
  637. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  638. Assert.Same(target.Panel.Children[9], result);
  639. }
  640. [Fact]
  641. public void GetControlInDirection_Down_Should_Scroll_If_Partially_Visible()
  642. {
  643. var target = CreateTarget();
  644. var scroller = (TestScroller)target.Parent;
  645. scroller.Width = 100;
  646. scroller.Height = 95;
  647. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  648. var from = target.Panel.Children[8];
  649. var result = ((ILogicalScrollable)target).GetControlInDirection(
  650. NavigationDirection.Down,
  651. from);
  652. Assert.Equal(new Vector(0, 1), ((ILogicalScrollable)target).Offset);
  653. Assert.Same(target.Panel.Children[8], result);
  654. }
  655. [Fact]
  656. public void GetControlInDirection_Up_Should_Scroll_If_Partially_Visible_Item_Is_Currently_Shown()
  657. {
  658. var target = CreateTarget();
  659. var scroller = (TestScroller)target.Parent;
  660. scroller.Width = 100;
  661. scroller.Height = 95;
  662. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  663. ((ILogicalScrollable)target).Offset = new Vector(0, 11);
  664. var from = target.Panel.Children[1];
  665. var result = ((ILogicalScrollable)target).GetControlInDirection(
  666. NavigationDirection.Up,
  667. from);
  668. Assert.Equal(new Vector(0, 10), ((ILogicalScrollable)target).Offset);
  669. Assert.Same(target.Panel.Children[0], result);
  670. }
  671. [Fact]
  672. public void Should_Return_Horizontal_Extent_And_Viewport()
  673. {
  674. var target = CreateTarget();
  675. target.ApplyTemplate();
  676. target.Measure(new Size(5, 100));
  677. target.Arrange(new Rect(0, 0, 5, 100));
  678. Assert.Equal(new Size(10, 20), ((ILogicalScrollable)target).Extent);
  679. Assert.Equal(new Size(5, 10), ((ILogicalScrollable)target).Viewport);
  680. }
  681. [Fact]
  682. public void Horizontal_Scroll_Should_Update_Item_Position()
  683. {
  684. var target = CreateTarget();
  685. target.ApplyTemplate();
  686. target.Measure(new Size(5, 100));
  687. target.Arrange(new Rect(0, 0, 5, 100));
  688. ((ILogicalScrollable)target).Offset = new Vector(5, 0);
  689. target.Measure(new Size(5, 100));
  690. target.Arrange(new Rect(0, 0, 5, 100));
  691. Assert.Equal(new Rect(-5, 0, 10, 10), target.Panel.Children[0].Bounds);
  692. }
  693. }
  694. public class Horizontal
  695. {
  696. [Fact]
  697. public void GetControlInDirection_Right_Should_Return_Existing_Container_If_Materialized()
  698. {
  699. var target = CreateTarget(orientation: Orientation.Horizontal);
  700. var scroller = (TestScroller)target.Parent;
  701. scroller.Width = scroller.Height = 100;
  702. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  703. var from = target.Panel.Children[5];
  704. var result = ((ILogicalScrollable)target).GetControlInDirection(
  705. NavigationDirection.Right,
  706. from);
  707. Assert.Same(target.Panel.Children[6], result);
  708. }
  709. [Fact]
  710. public void GetControlInDirection_Right_Should_Scroll_If_Necessary()
  711. {
  712. var target = CreateTarget(orientation: Orientation.Horizontal);
  713. var scroller = (TestScroller)target.Parent;
  714. scroller.Width = scroller.Height = 100;
  715. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  716. var from = target.Panel.Children[9];
  717. var result = ((ILogicalScrollable)target).GetControlInDirection(
  718. NavigationDirection.Right,
  719. from);
  720. Assert.Equal(new Vector(1, 0), ((ILogicalScrollable)target).Offset);
  721. Assert.Same(target.Panel.Children[9], result);
  722. }
  723. [Fact]
  724. public void GetControlInDirection_Right_Should_Scroll_If_Partially_Visible()
  725. {
  726. var target = CreateTarget(orientation: Orientation.Horizontal);
  727. var scroller = (TestScroller)target.Parent;
  728. scroller.Width = 95;
  729. scroller.Height = 100;
  730. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  731. var from = target.Panel.Children[8];
  732. var result = ((ILogicalScrollable)target).GetControlInDirection(
  733. NavigationDirection.Right,
  734. from);
  735. Assert.Equal(new Vector(1, 0), ((ILogicalScrollable)target).Offset);
  736. Assert.Same(target.Panel.Children[8], result);
  737. }
  738. [Fact]
  739. public void GetControlInDirection_Left_Should_Scroll_If_Partially_Visible_Item_Is_Currently_Shown()
  740. {
  741. var target = CreateTarget(orientation: Orientation.Horizontal);
  742. var scroller = (TestScroller)target.Parent;
  743. scroller.Width = 95;
  744. scroller.Height = 100;
  745. scroller.LayoutManager.ExecuteInitialLayoutPass(scroller);
  746. ((ILogicalScrollable)target).Offset = new Vector(11, 0);
  747. var from = target.Panel.Children[1];
  748. var result = ((ILogicalScrollable)target).GetControlInDirection(
  749. NavigationDirection.Left,
  750. from);
  751. Assert.Equal(new Vector(10, 0), ((ILogicalScrollable)target).Offset);
  752. Assert.Same(target.Panel.Children[0], result);
  753. }
  754. }
  755. private static ItemsPresenter CreateTarget(
  756. Orientation orientation = Orientation.Vertical,
  757. int itemCount = 20,
  758. bool useAvaloniaList = false)
  759. {
  760. ItemsPresenter result;
  761. var itemsSource = Enumerable.Range(0, itemCount).Select(x => $"Item {x}");
  762. var items = useAvaloniaList ?
  763. (IEnumerable)new AvaloniaList<string>(itemsSource) :
  764. (IEnumerable)new ObservableCollection<string>(itemsSource);
  765. var scroller = new TestScroller
  766. {
  767. CanHorizontallyScroll = true,
  768. CanVerticallyScroll = true,
  769. Content = result = new TestItemsPresenter
  770. {
  771. Items = items,
  772. ItemsPanel = VirtualizingPanelTemplate(orientation),
  773. DataTemplates = { StringDataTemplate() },
  774. VirtualizationMode = ItemVirtualizationMode.Simple,
  775. }
  776. };
  777. scroller.UpdateChild();
  778. return result;
  779. }
  780. private static IDataTemplate StringDataTemplate()
  781. {
  782. return new FuncDataTemplate<string>((x, _) => new Canvas
  783. {
  784. Width = 10,
  785. Height = 10,
  786. });
  787. }
  788. private static ITemplate<IPanel> VirtualizingPanelTemplate(
  789. Orientation orientation = Orientation.Vertical)
  790. {
  791. return new FuncTemplate<IPanel>(() => new VirtualizingStackPanel
  792. {
  793. Orientation = orientation,
  794. });
  795. }
  796. private class TestScroller : ScrollContentPresenter, IRenderRoot, ILayoutRoot, IStyleRoot
  797. {
  798. public IRenderer Renderer { get; }
  799. public Size ClientSize { get; }
  800. public double RenderScaling => 1;
  801. public Size MaxClientSize => Size.Infinity;
  802. public double LayoutScaling => 1;
  803. public ILayoutManager LayoutManager { get; } = new LayoutManager();
  804. public IRenderTarget CreateRenderTarget() => throw new NotImplementedException();
  805. public void Invalidate(Rect rect) => throw new NotImplementedException();
  806. public Point PointToClient(PixelPoint p) => throw new NotImplementedException();
  807. public PixelPoint PointToScreen(Point p) => throw new NotImplementedException();
  808. }
  809. private class TestItemsPresenter : ItemsPresenter
  810. {
  811. protected override IItemContainerGenerator CreateItemContainerGenerator()
  812. {
  813. return new ItemContainerGenerator<TestContainer>(
  814. this,
  815. TestContainer.ContentProperty,
  816. null);
  817. }
  818. }
  819. private class TestContainer : ContentControl
  820. {
  821. public TestContainer()
  822. {
  823. Template = new FuncControlTemplate<TestContainer>((parent, scope) => new ContentPresenter
  824. {
  825. Name = "PART_ContentPresenter",
  826. [~ContentPresenter.ContentProperty] = parent[~ContentControl.ContentProperty],
  827. [~ContentPresenter.ContentTemplateProperty] = parent[~ContentControl.ContentTemplateProperty],
  828. }.RegisterInNameScope(scope).WithNameScope(scope));
  829. }
  830. }
  831. }
  832. }