SelectingItemsControlTests_Multiple.cs 39 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195
  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.Presenters;
  10. using Avalonia.Controls.Primitives;
  11. using Avalonia.Controls.Templates;
  12. using Avalonia.Data;
  13. using Avalonia.Input;
  14. using Avalonia.Interactivity;
  15. using Avalonia.UnitTests;
  16. using Xunit;
  17. namespace Avalonia.Controls.UnitTests.Primitives
  18. {
  19. public class SelectingItemsControlTests_Multiple
  20. {
  21. private MouseTestHelper _helper = new MouseTestHelper();
  22. [Fact]
  23. public void Setting_SelectedIndex_Should_Add_To_SelectedItems()
  24. {
  25. var target = new TestSelector
  26. {
  27. Items = new[] { "foo", "bar" },
  28. Template = Template(),
  29. };
  30. target.ApplyTemplate();
  31. target.SelectedIndex = 1;
  32. Assert.Equal(new[] { "bar" }, target.SelectedItems.Cast<object>().ToList());
  33. }
  34. [Fact]
  35. public void Adding_SelectedItems_Should_Set_SelectedIndex()
  36. {
  37. var target = new TestSelector
  38. {
  39. Items = new[] { "foo", "bar" },
  40. Template = Template(),
  41. };
  42. target.ApplyTemplate();
  43. target.SelectedItems.Add("bar");
  44. Assert.Equal(1, target.SelectedIndex);
  45. }
  46. [Fact]
  47. public void Assigning_Single_SelectedItems_Should_Set_SelectedIndex()
  48. {
  49. var target = new TestSelector
  50. {
  51. Items = new[] { "foo", "bar" },
  52. Template = Template(),
  53. };
  54. target.ApplyTemplate();
  55. target.Presenter.ApplyTemplate();
  56. target.SelectedItems = new AvaloniaList<object>("bar");
  57. Assert.Equal(1, target.SelectedIndex);
  58. Assert.Equal(new[] { "bar" }, target.SelectedItems);
  59. Assert.Equal(new[] { 1 }, SelectedContainers(target));
  60. }
  61. [Fact]
  62. public void Assigning_Multiple_SelectedItems_Should_Set_SelectedIndex()
  63. {
  64. // Note that we don't need SelectionMode = Multiple here. Multiple selections can always
  65. // be made in code.
  66. var target = new TestSelector
  67. {
  68. Items = new[] { "foo", "bar", "baz" },
  69. Template = Template(),
  70. };
  71. target.ApplyTemplate();
  72. target.Presenter.ApplyTemplate();
  73. target.SelectedItems = new AvaloniaList<string>("foo", "bar", "baz");
  74. Assert.Equal(0, target.SelectedIndex);
  75. Assert.Equal(new[] { "foo", "bar", "baz" }, target.SelectedItems);
  76. Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target));
  77. }
  78. [Fact]
  79. public void Selected_Items_Should_Be_Marked_When_Panel_Created_After_SelectedItems_Is_Set()
  80. {
  81. // Issue #2565.
  82. var target = new TestSelector
  83. {
  84. Items = new[] { "foo", "bar", "baz" },
  85. Template = Template(),
  86. };
  87. target.ApplyTemplate();
  88. target.SelectedItems = new AvaloniaList<string>("foo", "bar", "baz");
  89. target.Presenter.ApplyTemplate();
  90. Assert.Equal(0, target.SelectedIndex);
  91. Assert.Equal(new[] { "foo", "bar", "baz" }, target.SelectedItems);
  92. Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target));
  93. }
  94. [Fact]
  95. public void Reassigning_SelectedItems_Should_Clear_Selection()
  96. {
  97. var target = new TestSelector
  98. {
  99. Items = new[] { "foo", "bar" },
  100. Template = Template(),
  101. };
  102. target.ApplyTemplate();
  103. target.SelectedItems.Add("bar");
  104. target.SelectedItems = new AvaloniaList<object>();
  105. Assert.Equal(-1, target.SelectedIndex);
  106. Assert.Null(target.SelectedItem);
  107. }
  108. [Fact]
  109. public void Adding_First_SelectedItem_Should_Raise_SelectedIndex_SelectedItem_Changed()
  110. {
  111. var target = new TestSelector
  112. {
  113. Items = new[] { "foo", "bar" },
  114. Template = Template(),
  115. };
  116. bool indexRaised = false;
  117. bool itemRaised = false;
  118. target.PropertyChanged += (s, e) =>
  119. {
  120. indexRaised |= e.Property.Name == "SelectedIndex" &&
  121. (int)e.OldValue == -1 &&
  122. (int)e.NewValue == 1;
  123. itemRaised |= e.Property.Name == "SelectedItem" &&
  124. (string)e.OldValue == null &&
  125. (string)e.NewValue == "bar";
  126. };
  127. target.ApplyTemplate();
  128. target.SelectedItems.Add("bar");
  129. Assert.True(indexRaised);
  130. Assert.True(itemRaised);
  131. }
  132. [Fact]
  133. public void Adding_Subsequent_SelectedItems_Should_Not_Raise_SelectedIndex_SelectedItem_Changed()
  134. {
  135. var target = new TestSelector
  136. {
  137. Items = new[] { "foo", "bar" },
  138. Template = Template(),
  139. };
  140. target.ApplyTemplate();
  141. target.SelectedItems.Add("foo");
  142. bool raised = false;
  143. target.PropertyChanged += (s, e) =>
  144. raised |= e.Property.Name == "SelectedIndex" ||
  145. e.Property.Name == "SelectedItem";
  146. target.SelectedItems.Add("bar");
  147. Assert.False(raised);
  148. }
  149. [Fact]
  150. public void Removing_Last_SelectedItem_Should_Raise_SelectedIndex_Changed()
  151. {
  152. var target = new TestSelector
  153. {
  154. Items = new[] { "foo", "bar" },
  155. Template = Template(),
  156. };
  157. target.ApplyTemplate();
  158. target.SelectedItems.Add("foo");
  159. bool raised = false;
  160. target.PropertyChanged += (s, e) =>
  161. raised |= e.Property.Name == "SelectedIndex" &&
  162. (int)e.OldValue == 0 &&
  163. (int)e.NewValue == -1;
  164. target.SelectedItems.RemoveAt(0);
  165. Assert.True(raised);
  166. }
  167. [Fact]
  168. public void Adding_SelectedItems_Should_Set_Item_IsSelected()
  169. {
  170. var items = new[]
  171. {
  172. new ListBoxItem(),
  173. new ListBoxItem(),
  174. new ListBoxItem(),
  175. };
  176. var target = new TestSelector
  177. {
  178. Items = items,
  179. Template = Template(),
  180. };
  181. target.ApplyTemplate();
  182. target.Presenter.ApplyTemplate();
  183. target.SelectedItems.Add(items[0]);
  184. target.SelectedItems.Add(items[1]);
  185. var foo = target.Presenter.Panel.Children[0];
  186. Assert.True(items[0].IsSelected);
  187. Assert.True(items[1].IsSelected);
  188. Assert.False(items[2].IsSelected);
  189. }
  190. [Fact]
  191. public void Assigning_SelectedItems_Should_Set_Item_IsSelected()
  192. {
  193. var items = new[]
  194. {
  195. new ListBoxItem(),
  196. new ListBoxItem(),
  197. new ListBoxItem(),
  198. };
  199. var target = new TestSelector
  200. {
  201. Items = items,
  202. Template = Template(),
  203. };
  204. target.ApplyTemplate();
  205. target.Presenter.ApplyTemplate();
  206. target.SelectedItems = new AvaloniaList<object> { items[0], items[1] };
  207. Assert.True(items[0].IsSelected);
  208. Assert.True(items[1].IsSelected);
  209. Assert.False(items[2].IsSelected);
  210. }
  211. [Fact]
  212. public void Removing_SelectedItems_Should_Clear_Item_IsSelected()
  213. {
  214. var items = new[]
  215. {
  216. new ListBoxItem(),
  217. new ListBoxItem(),
  218. new ListBoxItem(),
  219. };
  220. var target = new TestSelector
  221. {
  222. Items = items,
  223. Template = Template(),
  224. };
  225. target.ApplyTemplate();
  226. target.Presenter.ApplyTemplate();
  227. target.SelectedItems.Add(items[0]);
  228. target.SelectedItems.Add(items[1]);
  229. target.SelectedItems.Remove(items[1]);
  230. Assert.True(items[0].IsSelected);
  231. Assert.False(items[1].IsSelected);
  232. }
  233. [Fact]
  234. public void Reassigning_SelectedItems_Should_Clear_Item_IsSelected()
  235. {
  236. var items = new[]
  237. {
  238. new ListBoxItem(),
  239. new ListBoxItem(),
  240. new ListBoxItem(),
  241. };
  242. var target = new TestSelector
  243. {
  244. Items = items,
  245. Template = Template(),
  246. };
  247. target.ApplyTemplate();
  248. target.SelectedItems.Add(items[0]);
  249. target.SelectedItems.Add(items[1]);
  250. target.SelectedItems = new AvaloniaList<object> { items[0], items[1] };
  251. Assert.False(items[0].IsSelected);
  252. Assert.False(items[1].IsSelected);
  253. }
  254. [Fact]
  255. public void Setting_SelectedIndex_Should_Unmark_Previously_Selected_Containers()
  256. {
  257. var target = new TestSelector
  258. {
  259. Items = new[] { "foo", "bar", "baz" },
  260. Template = Template(),
  261. };
  262. target.ApplyTemplate();
  263. target.Presenter.ApplyTemplate();
  264. target.SelectedItems.Add("foo");
  265. target.SelectedItems.Add("bar");
  266. Assert.Equal(new[] { 0, 1 }, SelectedContainers(target));
  267. target.SelectedIndex = 2;
  268. Assert.Equal(new[] { 2 }, SelectedContainers(target));
  269. }
  270. [Fact]
  271. public void Range_Select_Should_Select_Range()
  272. {
  273. var target = new TestSelector
  274. {
  275. Items = new[]
  276. {
  277. "foo",
  278. "bar",
  279. "baz",
  280. "qux",
  281. "qiz",
  282. "lol",
  283. },
  284. SelectionMode = SelectionMode.Multiple,
  285. Template = Template(),
  286. };
  287. target.ApplyTemplate();
  288. target.SelectedIndex = 1;
  289. target.SelectRange(3);
  290. Assert.Equal(new[] { "bar", "baz", "qux" }, target.SelectedItems.Cast<object>().ToList());
  291. }
  292. [Fact]
  293. public void Range_Select_Backwards_Should_Select_Range()
  294. {
  295. var target = new TestSelector
  296. {
  297. Items = new[]
  298. {
  299. "foo",
  300. "bar",
  301. "baz",
  302. "qux",
  303. "qiz",
  304. "lol",
  305. },
  306. SelectionMode = SelectionMode.Multiple,
  307. Template = Template(),
  308. };
  309. target.ApplyTemplate();
  310. target.SelectedIndex = 3;
  311. target.SelectRange(1);
  312. Assert.Equal(new[] { "qux", "baz", "bar" }, target.SelectedItems.Cast<object>().ToList());
  313. }
  314. [Fact]
  315. public void Second_Range_Select_Backwards_Should_Select_From_Original_Selection()
  316. {
  317. var target = new TestSelector
  318. {
  319. Items = new[]
  320. {
  321. "foo",
  322. "bar",
  323. "baz",
  324. "qux",
  325. "qiz",
  326. "lol",
  327. },
  328. SelectionMode = SelectionMode.Multiple,
  329. Template = Template(),
  330. };
  331. target.ApplyTemplate();
  332. target.SelectedIndex = 2;
  333. target.SelectRange(5);
  334. target.SelectRange(4);
  335. Assert.Equal(new[] { "baz", "qux", "qiz" }, target.SelectedItems.Cast<object>().ToList());
  336. }
  337. [Fact]
  338. public void Setting_SelectedIndex_After_Range_Should_Unmark_Previously_Selected_Containers()
  339. {
  340. var target = new TestSelector
  341. {
  342. Items = new[] { "foo", "bar", "baz", "qux" },
  343. Template = Template(),
  344. SelectedIndex = 0,
  345. SelectionMode = SelectionMode.Multiple,
  346. };
  347. target.ApplyTemplate();
  348. target.Presenter.ApplyTemplate();
  349. target.SelectRange(2);
  350. Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target));
  351. target.SelectedIndex = 3;
  352. Assert.Equal(new[] { 3 }, SelectedContainers(target));
  353. }
  354. [Fact]
  355. public void Toggling_Selection_After_Range_Should_Work()
  356. {
  357. var target = new TestSelector
  358. {
  359. Items = new[] { "foo", "bar", "baz", "foo", "bar", "baz" },
  360. Template = Template(),
  361. SelectedIndex = 0,
  362. SelectionMode = SelectionMode.Multiple,
  363. };
  364. target.ApplyTemplate();
  365. target.Presenter.ApplyTemplate();
  366. target.SelectRange(3);
  367. Assert.Equal(new[] { 0, 1, 2, 3 }, SelectedContainers(target));
  368. target.Toggle(4);
  369. Assert.Equal(new[] { 0, 1, 2, 3, 4 }, SelectedContainers(target));
  370. }
  371. [Fact]
  372. public void Suprious_SelectedIndex_Changes_Should_Not_Be_Triggered()
  373. {
  374. var target = new TestSelector
  375. {
  376. Items = new[] { "foo", "bar", "baz" },
  377. Template = Template(),
  378. };
  379. target.ApplyTemplate();
  380. var selectedIndexes = new List<int>();
  381. target.GetObservable(TestSelector.SelectedIndexProperty).Subscribe(x => selectedIndexes.Add(x));
  382. target.SelectedItems = new AvaloniaList<object> { "bar", "baz" };
  383. target.SelectedItem = "foo";
  384. Assert.Equal(0, target.SelectedIndex);
  385. Assert.Equal(new[] { -1, 1, 0 }, selectedIndexes);
  386. }
  387. [Fact]
  388. public void Can_Set_SelectedIndex_To_Another_Selected_Item()
  389. {
  390. var target = new TestSelector
  391. {
  392. Items = new[] { "foo", "bar", "baz" },
  393. Template = Template(),
  394. };
  395. target.ApplyTemplate();
  396. target.Presenter.ApplyTemplate();
  397. target.SelectedItems.Add("foo");
  398. target.SelectedItems.Add("bar");
  399. Assert.Equal(0, target.SelectedIndex);
  400. Assert.Equal(new[] { "foo", "bar" }, target.SelectedItems);
  401. Assert.Equal(new[] { 0, 1 }, SelectedContainers(target));
  402. var raised = false;
  403. target.SelectionChanged += (s, e) =>
  404. {
  405. raised = true;
  406. Assert.Empty(e.AddedItems);
  407. Assert.Equal(new[] { "foo" }, e.RemovedItems);
  408. };
  409. target.SelectedIndex = 1;
  410. Assert.True(raised);
  411. Assert.Equal(1, target.SelectedIndex);
  412. Assert.Equal(new[] { "bar" }, target.SelectedItems);
  413. Assert.Equal(new[] { 1 }, SelectedContainers(target));
  414. }
  415. /// <summary>
  416. /// Tests a problem discovered with ListBox with selection.
  417. /// </summary>
  418. /// <remarks>
  419. /// - Items is bound to DataContext first, followed by say SelectedIndex
  420. /// - When the ListBox is removed from the visual tree, DataContext becomes null (as it's
  421. /// inherited)
  422. /// - This changes Items to null, which changes SelectedIndex to null as there are no
  423. /// longer any items
  424. /// - However, the news that DataContext is now null hasn't yet reached the SelectedItems
  425. /// binding and so the unselection is sent back to the ViewModel
  426. ///
  427. /// This is a similar problem to that tested by XamlBindingTest.Should_Not_Write_To_Old_DataContext.
  428. /// However, that tests a general property binding problem: here we are writing directly
  429. /// to the SelectedItems collection - not via a binding - so it's something that the
  430. /// binding system cannot solve. Instead we solve it by not clearing SelectedItems when
  431. /// DataContext is in the process of changing.
  432. /// </remarks>
  433. [Fact]
  434. public void Should_Not_Write_To_Old_DataContext()
  435. {
  436. var vm = new OldDataContextViewModel();
  437. var target = new TestSelector();
  438. var itemsBinding = new Binding
  439. {
  440. Path = "Items",
  441. Mode = BindingMode.OneWay,
  442. };
  443. var selectedItemsBinding = new Binding
  444. {
  445. Path = "SelectedItems",
  446. Mode = BindingMode.OneWay,
  447. };
  448. // Bind Items and SelectedItems to the VM.
  449. target.Bind(TestSelector.ItemsProperty, itemsBinding);
  450. target.Bind(TestSelector.SelectedItemsProperty, selectedItemsBinding);
  451. // Set DataContext and SelectedIndex
  452. target.DataContext = vm;
  453. target.SelectedIndex = 1;
  454. // Make sure SelectedItems are written back to VM.
  455. Assert.Equal(new[] { "bar" }, vm.SelectedItems);
  456. // Clear DataContext and ensure that SelectedItems is still set in the VM.
  457. target.DataContext = null;
  458. Assert.Equal(new[] { "bar" }, vm.SelectedItems);
  459. // Ensure target's SelectedItems is now clear.
  460. Assert.Empty(target.SelectedItems);
  461. }
  462. [Fact]
  463. public void Unbound_SelectedItems_Should_Be_Cleared_When_DataContext_Cleared()
  464. {
  465. var data = new
  466. {
  467. Items = new[] { "foo", "bar", "baz" },
  468. };
  469. var target = new TestSelector
  470. {
  471. DataContext = data,
  472. Template = Template(),
  473. };
  474. var itemsBinding = new Binding { Path = "Items" };
  475. target.Bind(TestSelector.ItemsProperty, itemsBinding);
  476. Assert.Same(data.Items, target.Items);
  477. target.SelectedItems.Add("bar");
  478. target.DataContext = null;
  479. Assert.Empty(target.SelectedItems);
  480. }
  481. [Fact]
  482. public void Adding_To_SelectedItems_Should_Raise_SelectionChanged()
  483. {
  484. var items = new[] { "foo", "bar", "baz" };
  485. var target = new TestSelector
  486. {
  487. DataContext = items,
  488. Template = Template(),
  489. Items = items,
  490. };
  491. var called = false;
  492. target.SelectionChanged += (s, e) =>
  493. {
  494. Assert.Equal(new[] { "bar" }, e.AddedItems.Cast<object>().ToList());
  495. Assert.Empty(e.RemovedItems);
  496. called = true;
  497. };
  498. target.SelectedItems.Add("bar");
  499. Assert.True(called);
  500. }
  501. [Fact]
  502. public void Removing_From_SelectedItems_Should_Raise_SelectionChanged()
  503. {
  504. var items = new[] { "foo", "bar", "baz" };
  505. var target = new TestSelector
  506. {
  507. Items = items,
  508. Template = Template(),
  509. SelectedItem = "bar",
  510. };
  511. var called = false;
  512. target.SelectionChanged += (s, e) =>
  513. {
  514. Assert.Equal(new[] { "bar" }, e.RemovedItems.Cast<object>().ToList());
  515. Assert.Empty(e.AddedItems);
  516. called = true;
  517. };
  518. target.SelectedItems.Remove("bar");
  519. Assert.True(called);
  520. }
  521. [Fact]
  522. public void Assigning_SelectedItems_Should_Raise_SelectionChanged()
  523. {
  524. var items = new[] { "foo", "bar", "baz" };
  525. var target = new TestSelector
  526. {
  527. Items = items,
  528. Template = Template(),
  529. SelectedItem = "bar",
  530. };
  531. var called = false;
  532. target.SelectionChanged += (s, e) =>
  533. {
  534. Assert.Equal(new[] { "foo", "baz" }, e.AddedItems.Cast<object>());
  535. Assert.Equal(new[] { "bar" }, e.RemovedItems.Cast<object>());
  536. called = true;
  537. };
  538. target.ApplyTemplate();
  539. target.Presenter.ApplyTemplate();
  540. target.SelectedItems = new AvaloniaList<object>("foo", "baz");
  541. Assert.True(called);
  542. }
  543. [Fact]
  544. public void Shift_Selecting_From_No_Selection_Selects_From_Start()
  545. {
  546. var target = new ListBox
  547. {
  548. Template = Template(),
  549. Items = new[] { "Foo", "Bar", "Baz" },
  550. SelectionMode = SelectionMode.Multiple,
  551. };
  552. target.ApplyTemplate();
  553. target.Presenter.ApplyTemplate();
  554. _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: InputModifiers.Shift);
  555. var panel = target.Presenter.Panel;
  556. Assert.Equal(new[] { "Foo", "Bar", "Baz" }, target.SelectedItems);
  557. Assert.Equal(new[] { 0, 1, 2 }, SelectedContainers(target));
  558. }
  559. [Fact]
  560. public void Ctrl_Selecting_SelectedItem_With_Multiple_Selection_Active_Sets_SelectedItem_To_Next_Selection()
  561. {
  562. var target = new ListBox
  563. {
  564. Template = Template(),
  565. Items = new[] { "Foo", "Bar", "Baz", "Qux" },
  566. SelectionMode = SelectionMode.Multiple,
  567. };
  568. target.ApplyTemplate();
  569. target.Presenter.ApplyTemplate();
  570. _helper.Click((Interactive)target.Presenter.Panel.Children[1]);
  571. _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: InputModifiers.Control);
  572. _helper.Click((Interactive)target.Presenter.Panel.Children[3], modifiers: InputModifiers.Control);
  573. Assert.Equal(1, target.SelectedIndex);
  574. Assert.Equal("Bar", target.SelectedItem);
  575. Assert.Equal(new[] { "Bar", "Baz", "Qux" }, target.SelectedItems);
  576. _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: InputModifiers.Control);
  577. Assert.Equal(2, target.SelectedIndex);
  578. Assert.Equal("Baz", target.SelectedItem);
  579. Assert.Equal(new[] { "Baz", "Qux" }, target.SelectedItems);
  580. }
  581. [Fact]
  582. public void Ctrl_Selecting_Non_SelectedItem_With_Multiple_Selection_Active_Leaves_SelectedItem_The_Same()
  583. {
  584. var target = new ListBox
  585. {
  586. Template = Template(),
  587. Items = new[] { "Foo", "Bar", "Baz" },
  588. SelectionMode = SelectionMode.Multiple,
  589. };
  590. target.ApplyTemplate();
  591. target.Presenter.ApplyTemplate();
  592. _helper.Click((Interactive)target.Presenter.Panel.Children[1]);
  593. _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: InputModifiers.Control);
  594. Assert.Equal(1, target.SelectedIndex);
  595. Assert.Equal("Bar", target.SelectedItem);
  596. _helper.Click((Interactive)target.Presenter.Panel.Children[2], modifiers: InputModifiers.Control);
  597. Assert.Equal(1, target.SelectedIndex);
  598. Assert.Equal("Bar", target.SelectedItem);
  599. }
  600. [Fact]
  601. public void Should_Ctrl_Select_Correct_Item_When_Duplicate_Items_Are_Present()
  602. {
  603. var target = new ListBox
  604. {
  605. Template = Template(),
  606. Items = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" },
  607. SelectionMode = SelectionMode.Multiple,
  608. };
  609. target.ApplyTemplate();
  610. target.Presenter.ApplyTemplate();
  611. _helper.Click((Interactive)target.Presenter.Panel.Children[3]);
  612. _helper.Click((Interactive)target.Presenter.Panel.Children[4], modifiers: InputModifiers.Control);
  613. var panel = target.Presenter.Panel;
  614. Assert.Equal(new[] { "Foo", "Bar" }, target.SelectedItems);
  615. Assert.Equal(new[] { 3, 4 }, SelectedContainers(target));
  616. }
  617. [Fact]
  618. public void Should_Shift_Select_Correct_Item_When_Duplicates_Are_Present()
  619. {
  620. var target = new ListBox
  621. {
  622. Template = Template(),
  623. Items = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" },
  624. SelectionMode = SelectionMode.Multiple,
  625. };
  626. target.ApplyTemplate();
  627. target.Presenter.ApplyTemplate();
  628. _helper.Click((Interactive)target.Presenter.Panel.Children[3]);
  629. _helper.Click((Interactive)target.Presenter.Panel.Children[5], modifiers: InputModifiers.Shift);
  630. var panel = target.Presenter.Panel;
  631. Assert.Equal(new[] { "Foo", "Bar", "Baz" }, target.SelectedItems);
  632. Assert.Equal(new[] { 3, 4, 5 }, SelectedContainers(target));
  633. }
  634. [Fact]
  635. public void Can_Shift_Select_All_Items_When_Duplicates_Are_Present()
  636. {
  637. var target = new ListBox
  638. {
  639. Template = Template(),
  640. Items = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" },
  641. SelectionMode = SelectionMode.Multiple,
  642. };
  643. target.ApplyTemplate();
  644. target.Presenter.ApplyTemplate();
  645. _helper.Click((Interactive)target.Presenter.Panel.Children[0]);
  646. _helper.Click((Interactive)target.Presenter.Panel.Children[5], modifiers: InputModifiers.Shift);
  647. var panel = target.Presenter.Panel;
  648. Assert.Equal(new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, target.SelectedItems);
  649. Assert.Equal(new[] { 0, 1, 2, 3, 4, 5 }, SelectedContainers(target));
  650. }
  651. [Fact]
  652. public void Duplicate_Items_Are_Added_To_SelectedItems_In_Order()
  653. {
  654. var target = new ListBox
  655. {
  656. Template = Template(),
  657. Items = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" },
  658. SelectionMode = SelectionMode.Multiple,
  659. };
  660. target.ApplyTemplate();
  661. target.Presenter.ApplyTemplate();
  662. _helper.Click((Interactive)target.Presenter.Panel.Children[0]);
  663. Assert.Equal(new[] { "Foo" }, target.SelectedItems);
  664. _helper.Click((Interactive)target.Presenter.Panel.Children[4], modifiers: InputModifiers.Control);
  665. Assert.Equal(new[] { "Foo", "Bar" }, target.SelectedItems);
  666. _helper.Click((Interactive)target.Presenter.Panel.Children[3], modifiers: InputModifiers.Control);
  667. Assert.Equal(new[] { "Foo", "Bar", "Foo" }, target.SelectedItems);
  668. _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: InputModifiers.Control);
  669. Assert.Equal(new[] { "Foo", "Bar", "Foo", "Bar" }, target.SelectedItems);
  670. }
  671. [Fact]
  672. public void SelectAll_Sets_SelectedIndex_And_SelectedItem()
  673. {
  674. var target = new TestSelector
  675. {
  676. Template = Template(),
  677. Items = new[] { "Foo", "Bar", "Baz" },
  678. SelectionMode = SelectionMode.Multiple,
  679. };
  680. target.ApplyTemplate();
  681. target.Presenter.ApplyTemplate();
  682. target.SelectAll();
  683. Assert.Equal(0, target.SelectedIndex);
  684. Assert.Equal("Foo", target.SelectedItem);
  685. }
  686. [Fact]
  687. public void UnselectAll_Clears_SelectedIndex_And_SelectedItem()
  688. {
  689. var target = new TestSelector
  690. {
  691. Template = Template(),
  692. Items = new[] { "Foo", "Bar", "Baz" },
  693. SelectionMode = SelectionMode.Multiple,
  694. SelectedIndex = 0,
  695. };
  696. target.ApplyTemplate();
  697. target.Presenter.ApplyTemplate();
  698. target.UnselectAll();
  699. Assert.Equal(-1, target.SelectedIndex);
  700. Assert.Equal(null, target.SelectedItem);
  701. }
  702. [Fact]
  703. public void SelectAll_Handles_Duplicate_Items()
  704. {
  705. var target = new TestSelector
  706. {
  707. Template = Template(),
  708. Items = new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" },
  709. SelectionMode = SelectionMode.Multiple,
  710. };
  711. target.ApplyTemplate();
  712. target.Presenter.ApplyTemplate();
  713. target.SelectAll();
  714. Assert.Equal(new[] { "Foo", "Bar", "Baz", "Foo", "Bar", "Baz" }, target.SelectedItems);
  715. }
  716. [Fact]
  717. public void Adding_Item_Before_SelectedItems_Should_Update_Selection()
  718. {
  719. var items = new ObservableCollection<string>
  720. {
  721. "Foo",
  722. "Bar",
  723. "Baz"
  724. };
  725. var target = new ListBox
  726. {
  727. Template = Template(),
  728. Items = items,
  729. SelectionMode = SelectionMode.Multiple,
  730. };
  731. target.ApplyTemplate();
  732. target.Presenter.ApplyTemplate();
  733. target.SelectAll();
  734. items.Insert(0, "Qux");
  735. Assert.Equal(1, target.SelectedIndex);
  736. Assert.Equal("Foo", target.SelectedItem);
  737. Assert.Equal(new[] { "Foo", "Bar", "Baz" }, target.SelectedItems);
  738. Assert.Equal(new[] { 1, 2, 3 }, SelectedContainers(target));
  739. }
  740. [Fact]
  741. public void Removing_Item_Before_SelectedItem_Should_Update_Selection()
  742. {
  743. var items = new ObservableCollection<string>
  744. {
  745. "Foo",
  746. "Bar",
  747. "Baz"
  748. };
  749. var target = new TestSelector
  750. {
  751. Template = Template(),
  752. Items = items,
  753. SelectionMode = SelectionMode.Multiple,
  754. };
  755. target.ApplyTemplate();
  756. target.Presenter.ApplyTemplate();
  757. target.SelectedIndex = 1;
  758. target.SelectRange(2);
  759. Assert.Equal(new[] { "Bar", "Baz" }, target.SelectedItems);
  760. items.RemoveAt(0);
  761. Assert.Equal(0, target.SelectedIndex);
  762. Assert.Equal("Bar", target.SelectedItem);
  763. Assert.Equal(new[] { "Bar", "Baz" }, target.SelectedItems);
  764. Assert.Equal(new[] { 0, 1 }, SelectedContainers(target));
  765. }
  766. [Fact]
  767. public void Removing_SelectedItem_With_Multiple_Selection_Active_Should_Update_Selection()
  768. {
  769. var items = new ObservableCollection<string>
  770. {
  771. "Foo",
  772. "Bar",
  773. "Baz"
  774. };
  775. var target = new ListBox
  776. {
  777. Template = Template(),
  778. Items = items,
  779. SelectionMode = SelectionMode.Multiple,
  780. };
  781. target.ApplyTemplate();
  782. target.Presenter.ApplyTemplate();
  783. target.SelectAll();
  784. items.RemoveAt(0);
  785. Assert.Equal(0, target.SelectedIndex);
  786. Assert.Equal("Bar", target.SelectedItem);
  787. Assert.Equal(new[] { "Bar", "Baz" }, target.SelectedItems);
  788. Assert.Equal(new[] { 0, 1 }, SelectedContainers(target));
  789. }
  790. [Fact]
  791. public void Replacing_Selected_Item_Should_Update_SelectedItems()
  792. {
  793. var items = new ObservableCollection<string>
  794. {
  795. "Foo",
  796. "Bar",
  797. "Baz"
  798. };
  799. var target = new ListBox
  800. {
  801. Template = Template(),
  802. Items = items,
  803. SelectionMode = SelectionMode.Multiple,
  804. };
  805. target.ApplyTemplate();
  806. target.Presenter.ApplyTemplate();
  807. target.SelectAll();
  808. items[1] = "Qux";
  809. Assert.Equal(new[] { "Foo", "Qux", "Baz" }, target.SelectedItems);
  810. }
  811. [Fact]
  812. public void Left_Click_On_SelectedItem_Should_Clear_Existing_Selection()
  813. {
  814. var target = new ListBox
  815. {
  816. Template = Template(),
  817. Items = new[] { "Foo", "Bar", "Baz" },
  818. ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
  819. SelectionMode = SelectionMode.Multiple,
  820. };
  821. target.ApplyTemplate();
  822. target.Presenter.ApplyTemplate();
  823. target.SelectAll();
  824. Assert.Equal(3, target.SelectedItems.Count);
  825. _helper.Click((Interactive)target.Presenter.Panel.Children[0]);
  826. Assert.Equal(1, target.SelectedItems.Count);
  827. Assert.Equal(new[] { "Foo", }, target.SelectedItems);
  828. Assert.Equal(new[] { 0 }, SelectedContainers(target));
  829. }
  830. [Fact]
  831. public void Right_Click_On_SelectedItem_Should_Not_Clear_Existing_Selection()
  832. {
  833. var target = new ListBox
  834. {
  835. Template = Template(),
  836. Items = new[] { "Foo", "Bar", "Baz" },
  837. ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
  838. SelectionMode = SelectionMode.Multiple,
  839. };
  840. target.ApplyTemplate();
  841. target.Presenter.ApplyTemplate();
  842. target.SelectAll();
  843. Assert.Equal(3, target.SelectedItems.Count);
  844. _helper.Click((Interactive)target.Presenter.Panel.Children[0], MouseButton.Right);
  845. Assert.Equal(3, target.SelectedItems.Count);
  846. }
  847. [Fact]
  848. public void Right_Click_On_UnselectedItem_Should_Clear_Existing_Selection()
  849. {
  850. var target = new ListBox
  851. {
  852. Template = Template(),
  853. Items = new[] { "Foo", "Bar", "Baz" },
  854. ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
  855. SelectionMode = SelectionMode.Multiple,
  856. };
  857. target.ApplyTemplate();
  858. target.Presenter.ApplyTemplate();
  859. _helper.Click((Interactive)target.Presenter.Panel.Children[0]);
  860. _helper.Click((Interactive)target.Presenter.Panel.Children[1], modifiers: InputModifiers.Shift);
  861. Assert.Equal(2, target.SelectedItems.Count);
  862. _helper.Click((Interactive)target.Presenter.Panel.Children[2], MouseButton.Right);
  863. Assert.Equal(1, target.SelectedItems.Count);
  864. }
  865. [Fact]
  866. public void Adding_Selected_ItemContainers_Should_Update_Selection()
  867. {
  868. var items = new AvaloniaList<ItemContainer>(new[]
  869. {
  870. new ItemContainer(),
  871. new ItemContainer(),
  872. });
  873. var target = new TestSelector
  874. {
  875. Items = items,
  876. Template = Template(),
  877. };
  878. target.ApplyTemplate();
  879. target.Presenter.ApplyTemplate();
  880. items.Add(new ItemContainer { IsSelected = true });
  881. items.Add(new ItemContainer { IsSelected = true });
  882. Assert.Equal(2, target.SelectedIndex);
  883. Assert.Equal(items[2], target.SelectedItem);
  884. Assert.Equal(new[] { items[2], items[3] }, target.SelectedItems);
  885. }
  886. [Fact]
  887. public void Shift_Right_Click_Should_Not_Select_Multiple()
  888. {
  889. var target = new ListBox
  890. {
  891. Template = Template(),
  892. Items = new[] { "Foo", "Bar", "Baz" },
  893. ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
  894. SelectionMode = SelectionMode.Multiple,
  895. };
  896. target.ApplyTemplate();
  897. target.Presenter.ApplyTemplate();
  898. _helper.Click((Interactive)target.Presenter.Panel.Children[0]);
  899. _helper.Click((Interactive)target.Presenter.Panel.Children[2], MouseButton.Right, modifiers: InputModifiers.Shift);
  900. Assert.Equal(1, target.SelectedItems.Count);
  901. }
  902. [Fact]
  903. public void Ctrl_Right_Click_Should_Not_Select_Multiple()
  904. {
  905. var target = new ListBox
  906. {
  907. Template = Template(),
  908. Items = new[] { "Foo", "Bar", "Baz" },
  909. ItemTemplate = new FuncDataTemplate<string>((x, _) => new TextBlock { Width = 20, Height = 10 }),
  910. SelectionMode = SelectionMode.Multiple,
  911. };
  912. target.ApplyTemplate();
  913. target.Presenter.ApplyTemplate();
  914. _helper.Click((Interactive)target.Presenter.Panel.Children[0]);
  915. _helper.Click((Interactive)target.Presenter.Panel.Children[2], MouseButton.Right, modifiers: InputModifiers.Control);
  916. Assert.Equal(1, target.SelectedItems.Count);
  917. }
  918. private IEnumerable<int> SelectedContainers(SelectingItemsControl target)
  919. {
  920. return target.Presenter.Panel.Children
  921. .Select((x, i) => x.Classes.Contains(":selected") ? i : -1)
  922. .Where(x => x != -1);
  923. }
  924. private FuncControlTemplate Template()
  925. {
  926. return new FuncControlTemplate<SelectingItemsControl>((control, scope) =>
  927. new ItemsPresenter
  928. {
  929. Name = "PART_ItemsPresenter",
  930. [~ItemsPresenter.ItemsProperty] = control[~ItemsControl.ItemsProperty],
  931. [~ItemsPresenter.ItemsPanelProperty] = control[~ItemsControl.ItemsPanelProperty],
  932. }.RegisterInNameScope(scope));
  933. }
  934. private class TestSelector : SelectingItemsControl
  935. {
  936. public static readonly new AvaloniaProperty<IList> SelectedItemsProperty =
  937. SelectingItemsControl.SelectedItemsProperty;
  938. public new IList SelectedItems
  939. {
  940. get { return base.SelectedItems; }
  941. set { base.SelectedItems = value; }
  942. }
  943. public new SelectionMode SelectionMode
  944. {
  945. get { return base.SelectionMode; }
  946. set { base.SelectionMode = value; }
  947. }
  948. public new void SelectAll() => base.SelectAll();
  949. public new void UnselectAll() => base.UnselectAll();
  950. public void SelectRange(int index) => UpdateSelection(index, true, true);
  951. public void Toggle(int index) => UpdateSelection(index, true, false, true);
  952. }
  953. private class OldDataContextViewModel
  954. {
  955. public OldDataContextViewModel()
  956. {
  957. Items = new List<string> { "foo", "bar" };
  958. SelectedItems = new List<string>();
  959. }
  960. public List<string> Items { get; }
  961. public List<string> SelectedItems { get; }
  962. }
  963. private class ItemContainer : Control, ISelectable
  964. {
  965. public string Value { get; set; }
  966. public bool IsSelected { get; set; }
  967. }
  968. }
  969. }