ListViewTests.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663
  1. using System.Collections;
  2. using Xunit.Abstractions;
  3. namespace Terminal.Gui.ViewsTests;
  4. public class ListViewTests
  5. {
  6. private readonly ITestOutputHelper _output;
  7. public ListViewTests (ITestOutputHelper output) { _output = output; }
  8. [Fact]
  9. public void Constructors_Defaults ()
  10. {
  11. var lv = new ListView ();
  12. Assert.Null (lv.Source);
  13. Assert.True (lv.CanFocus);
  14. Assert.Equal (-1, lv.SelectedItem);
  15. lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two", "Three" }) };
  16. Assert.NotNull (lv.Source);
  17. Assert.Equal (-1, lv.SelectedItem);
  18. lv = new ListView { Source = new NewListDataSource () };
  19. Assert.NotNull (lv.Source);
  20. Assert.Equal (-1, lv.SelectedItem);
  21. lv = new ListView
  22. {
  23. Y = 1, Width = 10, Height = 20, Source = new ListWrapper (new List<string> { "One", "Two", "Three" })
  24. };
  25. Assert.NotNull (lv.Source);
  26. Assert.Equal (-1, lv.SelectedItem);
  27. Assert.Equal (new Rect (0, 1, 10, 20), lv.Frame);
  28. lv = new ListView { Y = 1, Width = 10, Height = 20, Source = new NewListDataSource () };
  29. Assert.NotNull (lv.Source);
  30. Assert.Equal (-1, lv.SelectedItem);
  31. Assert.Equal (new Rect (0, 1, 10, 20), lv.Frame);
  32. }
  33. [Fact]
  34. [AutoInitShutdown]
  35. public void Ensures_Visibility_SelectedItem_On_MoveDown_And_MoveUp ()
  36. {
  37. List<string> source = new ();
  38. for (var i = 0; i < 20; i++)
  39. {
  40. source.Add ($"Line{i}");
  41. }
  42. var lv = new ListView { Width = Dim.Fill (), Height = Dim.Fill (), Source = new ListWrapper (source) };
  43. var win = new Window ();
  44. win.Add (lv);
  45. Application.Top.Add (win);
  46. Application.Begin (Application.Top);
  47. ((FakeDriver)Application.Driver).SetBufferSize (12, 12);
  48. Application.Refresh ();
  49. Assert.Equal (-1, lv.SelectedItem);
  50. TestHelpers.AssertDriverContentsWithFrameAre (
  51. @"
  52. ┌──────────┐
  53. │Line0 │
  54. │Line1 │
  55. │Line2 │
  56. │Line3 │
  57. │Line4 │
  58. │Line5 │
  59. │Line6 │
  60. │Line7 │
  61. │Line8 │
  62. │Line9 │
  63. └──────────┘",
  64. _output
  65. );
  66. Assert.True (lv.ScrollDown (10));
  67. lv.Draw ();
  68. Assert.Equal (-1, lv.SelectedItem);
  69. TestHelpers.AssertDriverContentsWithFrameAre (
  70. @"
  71. ┌──────────┐
  72. │Line10 │
  73. │Line11 │
  74. │Line12 │
  75. │Line13 │
  76. │Line14 │
  77. │Line15 │
  78. │Line16 │
  79. │Line17 │
  80. │Line18 │
  81. │Line19 │
  82. └──────────┘",
  83. _output
  84. );
  85. Assert.True (lv.MoveDown ());
  86. lv.Draw ();
  87. Assert.Equal (0, lv.SelectedItem);
  88. TestHelpers.AssertDriverContentsWithFrameAre (
  89. @"
  90. ┌──────────┐
  91. │Line0 │
  92. │Line1 │
  93. │Line2 │
  94. │Line3 │
  95. │Line4 │
  96. │Line5 │
  97. │Line6 │
  98. │Line7 │
  99. │Line8 │
  100. │Line9 │
  101. └──────────┘",
  102. _output
  103. );
  104. Assert.True (lv.MoveEnd ());
  105. lv.Draw ();
  106. Assert.Equal (19, lv.SelectedItem);
  107. TestHelpers.AssertDriverContentsWithFrameAre (
  108. @"
  109. ┌──────────┐
  110. │Line19 │
  111. │ │
  112. │ │
  113. │ │
  114. │ │
  115. │ │
  116. │ │
  117. │ │
  118. │ │
  119. │ │
  120. └──────────┘",
  121. _output
  122. );
  123. Assert.True (lv.ScrollUp (20));
  124. lv.Draw ();
  125. Assert.Equal (19, lv.SelectedItem);
  126. TestHelpers.AssertDriverContentsWithFrameAre (
  127. @"
  128. ┌──────────┐
  129. │Line0 │
  130. │Line1 │
  131. │Line2 │
  132. │Line3 │
  133. │Line4 │
  134. │Line5 │
  135. │Line6 │
  136. │Line7 │
  137. │Line8 │
  138. │Line9 │
  139. └──────────┘",
  140. _output
  141. );
  142. Assert.True (lv.MoveDown ());
  143. lv.Draw ();
  144. Assert.Equal (19, lv.SelectedItem);
  145. TestHelpers.AssertDriverContentsWithFrameAre (
  146. @"
  147. ┌──────────┐
  148. │Line10 │
  149. │Line11 │
  150. │Line12 │
  151. │Line13 │
  152. │Line14 │
  153. │Line15 │
  154. │Line16 │
  155. │Line17 │
  156. │Line18 │
  157. │Line19 │
  158. └──────────┘",
  159. _output
  160. );
  161. Assert.True (lv.ScrollUp (20));
  162. lv.Draw ();
  163. Assert.Equal (19, lv.SelectedItem);
  164. TestHelpers.AssertDriverContentsWithFrameAre (
  165. @"
  166. ┌──────────┐
  167. │Line0 │
  168. │Line1 │
  169. │Line2 │
  170. │Line3 │
  171. │Line4 │
  172. │Line5 │
  173. │Line6 │
  174. │Line7 │
  175. │Line8 │
  176. │Line9 │
  177. └──────────┘",
  178. _output
  179. );
  180. Assert.True (lv.MoveDown ());
  181. lv.Draw ();
  182. Assert.Equal (19, lv.SelectedItem);
  183. TestHelpers.AssertDriverContentsWithFrameAre (
  184. @"
  185. ┌──────────┐
  186. │Line10 │
  187. │Line11 │
  188. │Line12 │
  189. │Line13 │
  190. │Line14 │
  191. │Line15 │
  192. │Line16 │
  193. │Line17 │
  194. │Line18 │
  195. │Line19 │
  196. └──────────┘",
  197. _output
  198. );
  199. Assert.True (lv.MoveHome ());
  200. lv.Draw ();
  201. Assert.Equal (0, lv.SelectedItem);
  202. TestHelpers.AssertDriverContentsWithFrameAre (
  203. @"
  204. ┌──────────┐
  205. │Line0 │
  206. │Line1 │
  207. │Line2 │
  208. │Line3 │
  209. │Line4 │
  210. │Line5 │
  211. │Line6 │
  212. │Line7 │
  213. │Line8 │
  214. │Line9 │
  215. └──────────┘",
  216. _output
  217. );
  218. Assert.True (lv.ScrollDown (20));
  219. lv.Draw ();
  220. Assert.Equal (0, lv.SelectedItem);
  221. TestHelpers.AssertDriverContentsWithFrameAre (
  222. @"
  223. ┌──────────┐
  224. │Line19 │
  225. │ │
  226. │ │
  227. │ │
  228. │ │
  229. │ │
  230. │ │
  231. │ │
  232. │ │
  233. │ │
  234. └──────────┘",
  235. _output
  236. );
  237. Assert.True (lv.MoveUp ());
  238. lv.Draw ();
  239. Assert.Equal (0, lv.SelectedItem);
  240. TestHelpers.AssertDriverContentsWithFrameAre (
  241. @"
  242. ┌──────────┐
  243. │Line0 │
  244. │Line1 │
  245. │Line2 │
  246. │Line3 │
  247. │Line4 │
  248. │Line5 │
  249. │Line6 │
  250. │Line7 │
  251. │Line8 │
  252. │Line9 │
  253. └──────────┘",
  254. _output
  255. );
  256. }
  257. [Fact]
  258. [AutoInitShutdown]
  259. public void EnsureSelectedItemVisible_SelectedItem ()
  260. {
  261. List<string> source = new ();
  262. for (var i = 0; i < 10; i++)
  263. {
  264. source.Add ($"Item {i}");
  265. }
  266. var lv = new ListView { Width = 10, Height = 5, Source = new ListWrapper (source) };
  267. Application.Top.Add (lv);
  268. Application.Begin (Application.Top);
  269. TestHelpers.AssertDriverContentsWithFrameAre (
  270. @"
  271. Item 0
  272. Item 1
  273. Item 2
  274. Item 3
  275. Item 4",
  276. _output
  277. );
  278. // EnsureSelectedItemVisible is auto enabled on the OnSelectedChanged
  279. lv.SelectedItem = 6;
  280. Application.Refresh ();
  281. TestHelpers.AssertDriverContentsWithFrameAre (
  282. @"
  283. Item 2
  284. Item 3
  285. Item 4
  286. Item 5
  287. Item 6",
  288. _output
  289. );
  290. }
  291. [Fact]
  292. [AutoInitShutdown]
  293. public void EnsureSelectedItemVisible_Top ()
  294. {
  295. List<string> source = new () { "First", "Second" };
  296. var lv = new ListView { Width = Dim.Fill (), Height = 1, Source = new ListWrapper (source) };
  297. lv.SelectedItem = 1;
  298. Application.Top.Add (lv);
  299. Application.Begin (Application.Top);
  300. Assert.Equal ("Second ", GetContents (0));
  301. Assert.Equal (new string (' ', 7), GetContents (1));
  302. lv.MoveUp ();
  303. lv.Draw ();
  304. Assert.Equal ("First ", GetContents (0));
  305. Assert.Equal (new string (' ', 7), GetContents (1));
  306. string GetContents (int line)
  307. {
  308. var item = "";
  309. for (var i = 0; i < 7; i++)
  310. {
  311. item += Application.Driver.Contents [line, i].Rune;
  312. }
  313. return item;
  314. }
  315. }
  316. [Fact]
  317. public void KeyBindings_Command ()
  318. {
  319. List<string> source = new () { "One", "Two", "Three" };
  320. var lv = new ListView { Height = 2, AllowsMarking = true, Source = new ListWrapper (source) };
  321. lv.BeginInit ();
  322. lv.EndInit ();
  323. Assert.Equal (-1, lv.SelectedItem);
  324. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.CursorDown)));
  325. Assert.Equal (0, lv.SelectedItem);
  326. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.CursorUp)));
  327. Assert.Equal (0, lv.SelectedItem);
  328. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.PageDown)));
  329. Assert.Equal (2, lv.SelectedItem);
  330. Assert.Equal (2, lv.TopItem);
  331. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.PageUp)));
  332. Assert.Equal (0, lv.SelectedItem);
  333. Assert.Equal (0, lv.TopItem);
  334. Assert.False (lv.Source.IsMarked (lv.SelectedItem));
  335. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.Space)));
  336. Assert.True (lv.Source.IsMarked (lv.SelectedItem));
  337. var opened = false;
  338. lv.OpenSelectedItem += (s, _) => opened = true;
  339. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.Enter)));
  340. Assert.True (opened);
  341. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.End)));
  342. Assert.Equal (2, lv.SelectedItem);
  343. Assert.True (lv.NewKeyDownEvent (new Key (KeyCode.Home)));
  344. Assert.Equal (0, lv.SelectedItem);
  345. }
  346. /// <summary>
  347. /// Tests that when none of the Commands in a chained keybinding are possible the
  348. /// <see cref="View.NewKeyDownEvent"/> returns the appropriate result
  349. /// </summary>
  350. [Fact]
  351. public void ListViewProcessKeyReturnValue_WithMultipleCommands ()
  352. {
  353. var lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two", "Three", "Four" }) };
  354. Assert.NotNull (lv.Source);
  355. // first item should be deselected by default
  356. Assert.Equal (-1, lv.SelectedItem);
  357. // bind shift down to move down twice in control
  358. lv.KeyBindings.Add (KeyCode.CursorDown | KeyCode.ShiftMask, Command.LineDown, Command.LineDown);
  359. var ev = new Key (KeyCode.CursorDown | KeyCode.ShiftMask);
  360. Assert.True (lv.NewKeyDownEvent (ev), "The first time we move down 2 it should be possible");
  361. // After moving down twice from -1 we should be at 'Two'
  362. Assert.Equal (1, lv.SelectedItem);
  363. // clear the items
  364. lv.SetSource (null);
  365. // Press key combo again - return should be false this time as none of the Commands are allowable
  366. Assert.False (lv.NewKeyDownEvent (ev), "We cannot move down so will not respond to this");
  367. }
  368. [Fact]
  369. public void ListViewSelectThenDown ()
  370. {
  371. var lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two", "Three" }) };
  372. lv.AllowsMarking = true;
  373. Assert.NotNull (lv.Source);
  374. // first item should be deselected by default
  375. Assert.Equal (-1, lv.SelectedItem);
  376. // nothing is ticked
  377. Assert.False (lv.Source.IsMarked (0));
  378. Assert.False (lv.Source.IsMarked (1));
  379. Assert.False (lv.Source.IsMarked (2));
  380. lv.KeyBindings.Add (KeyCode.Space | KeyCode.ShiftMask, Command.ToggleChecked, Command.LineDown);
  381. var ev = new Key (KeyCode.Space | KeyCode.ShiftMask);
  382. // view should indicate that it has accepted and consumed the event
  383. Assert.True (lv.NewKeyDownEvent (ev));
  384. // first item should now be selected
  385. Assert.Equal (0, lv.SelectedItem);
  386. // none of the items should be ticked
  387. Assert.False (lv.Source.IsMarked (0));
  388. Assert.False (lv.Source.IsMarked (1));
  389. Assert.False (lv.Source.IsMarked (2));
  390. // Press key combo again
  391. Assert.True (lv.NewKeyDownEvent (ev));
  392. // second item should now be selected
  393. Assert.Equal (1, lv.SelectedItem);
  394. // first item only should be ticked
  395. Assert.True (lv.Source.IsMarked (0));
  396. Assert.False (lv.Source.IsMarked (1));
  397. Assert.False (lv.Source.IsMarked (2));
  398. // Press key combo again
  399. Assert.True (lv.NewKeyDownEvent (ev));
  400. Assert.Equal (2, lv.SelectedItem);
  401. Assert.True (lv.Source.IsMarked (0));
  402. Assert.True (lv.Source.IsMarked (1));
  403. Assert.False (lv.Source.IsMarked (2));
  404. // Press key combo again
  405. Assert.True (lv.NewKeyDownEvent (ev));
  406. Assert.Equal (2, lv.SelectedItem); // cannot move down any further
  407. Assert.True (lv.Source.IsMarked (0));
  408. Assert.True (lv.Source.IsMarked (1));
  409. Assert.True (lv.Source.IsMarked (2)); // but can toggle marked
  410. // Press key combo again
  411. Assert.True (lv.NewKeyDownEvent (ev));
  412. Assert.Equal (2, lv.SelectedItem); // cannot move down any further
  413. Assert.True (lv.Source.IsMarked (0));
  414. Assert.True (lv.Source.IsMarked (1));
  415. Assert.False (lv.Source.IsMarked (2)); // untoggle toggle marked
  416. }
  417. [Fact]
  418. public void ListWrapper_StartsWith ()
  419. {
  420. var lw = new ListWrapper (new List<string> { "One", "Two", "Three" });
  421. Assert.Equal (1, lw.StartsWith ("t"));
  422. Assert.Equal (1, lw.StartsWith ("tw"));
  423. Assert.Equal (2, lw.StartsWith ("th"));
  424. Assert.Equal (1, lw.StartsWith ("T"));
  425. Assert.Equal (1, lw.StartsWith ("TW"));
  426. Assert.Equal (2, lw.StartsWith ("TH"));
  427. lw = new ListWrapper (new List<string> { "One", "Two", "Three" });
  428. Assert.Equal (1, lw.StartsWith ("t"));
  429. Assert.Equal (1, lw.StartsWith ("tw"));
  430. Assert.Equal (2, lw.StartsWith ("th"));
  431. Assert.Equal (1, lw.StartsWith ("T"));
  432. Assert.Equal (1, lw.StartsWith ("TW"));
  433. Assert.Equal (2, lw.StartsWith ("TH"));
  434. }
  435. [Fact]
  436. public void OnEnter_Does_Not_Throw_Exception ()
  437. {
  438. var lv = new ListView ();
  439. var top = new View ();
  440. top.Add (lv);
  441. Exception exception = Record.Exception (lv.SetFocus);
  442. Assert.Null (exception);
  443. }
  444. [Fact]
  445. [AutoInitShutdown]
  446. public void RowRender_Event ()
  447. {
  448. var rendered = false;
  449. List<string> source = new () { "one", "two", "three" };
  450. var lv = new ListView { Width = Dim.Fill (), Height = Dim.Fill () };
  451. lv.RowRender += (s, _) => rendered = true;
  452. Application.Top.Add (lv);
  453. Application.Begin (Application.Top);
  454. Assert.False (rendered);
  455. lv.SetSource (source);
  456. lv.Draw ();
  457. Assert.True (rendered);
  458. }
  459. [Fact]
  460. public void SelectedItem_Get_Set ()
  461. {
  462. var lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two", "Three" }) };
  463. Assert.Equal (-1, lv.SelectedItem);
  464. Assert.Throws<ArgumentException> (() => lv.SelectedItem = 3);
  465. Exception exception = Record.Exception (() => lv.SelectedItem = -1);
  466. Assert.Null (exception);
  467. }
  468. [Fact]
  469. public void SetSource_Preserves_ListWrapper_Instance_If_Not_Null ()
  470. {
  471. var lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two" }) };
  472. Assert.NotNull (lv.Source);
  473. lv.SetSource (null);
  474. Assert.NotNull (lv.Source);
  475. lv.Source = null;
  476. Assert.Null (lv.Source);
  477. lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two" }) };
  478. Assert.NotNull (lv.Source);
  479. lv.SetSourceAsync (null);
  480. Assert.NotNull (lv.Source);
  481. }
  482. [Fact]
  483. public void SettingEmptyKeybindingThrows ()
  484. {
  485. var lv = new ListView { Source = new ListWrapper (new List<string> { "One", "Two", "Three" }) };
  486. Assert.Throws<ArgumentException> (() => lv.KeyBindings.Add (KeyCode.Space));
  487. }
  488. private class NewListDataSource : IListDataSource
  489. {
  490. public int Count => throw new NotImplementedException ();
  491. public int Length => throw new NotImplementedException ();
  492. public bool IsMarked (int item) { throw new NotImplementedException (); }
  493. public void Render (
  494. ListView container,
  495. ConsoleDriver driver,
  496. bool selected,
  497. int item,
  498. int col,
  499. int line,
  500. int width,
  501. int start = 0
  502. )
  503. {
  504. throw new NotImplementedException ();
  505. }
  506. public void SetMark (int item, bool value) { throw new NotImplementedException (); }
  507. public IList ToList () { return new List<string> { "One", "Two", "Three" }; }
  508. }
  509. // No longer needed given PR #2920
  510. // [Fact, AutoInitShutdown]
  511. // public void Clicking_On_Border_Is_Ignored ()
  512. // {
  513. // var selected = "";
  514. // var lv = new ListView {
  515. // Height = 5,
  516. // Width = 7,
  517. // BorderStyle = LineStyle.Single
  518. // };
  519. // lv.SetSource (new List<string> { "One", "Two", "Three", "Four" });
  520. // lv.SelectedItemChanged += (s, e) => selected = e.Value.ToString ();
  521. // Application.Top.Add (lv);
  522. // Application.Begin (Application.Top);
  523. // Assert.Equal (new Thickness (1), lv.Border.Thickness);
  524. // Assert.Equal (-1, lv.SelectedItem);
  525. // Assert.Equal ("", lv.Text);
  526. // TestHelpers.AssertDriverContentsWithFrameAre (@"
  527. //┌─────┐
  528. //│One │
  529. //│Two │
  530. //│Three│
  531. //└─────┘", output);
  532. // Assert.True (lv.MouseEvent (new MouseEvent {
  533. // X = 0,
  534. // Y = 0,
  535. // Flags = MouseFlags.Button1Clicked
  536. // }));
  537. // Assert.Equal ("", selected);
  538. // Assert.Equal (-1, lv.SelectedItem);
  539. // Assert.True (lv.MouseEvent (new MouseEvent {
  540. // X = 0,
  541. // Y = 1,
  542. // Flags = MouseFlags.Button1Clicked
  543. // }));
  544. // Assert.Equal ("One", selected);
  545. // Assert.Equal (0, lv.SelectedItem);
  546. // Assert.True (lv.MouseEvent (new MouseEvent {
  547. // X = 0,
  548. // Y = 2,
  549. // Flags = MouseFlags.Button1Clicked
  550. // }));
  551. // Assert.Equal ("Two", selected);
  552. // Assert.Equal (1, lv.SelectedItem);
  553. // Assert.True (lv.MouseEvent (new MouseEvent {
  554. // X = 0,
  555. // Y = 3,
  556. // Flags = MouseFlags.Button1Clicked
  557. // }));
  558. // Assert.Equal ("Three", selected);
  559. // Assert.Equal (2, lv.SelectedItem);
  560. // Assert.True (lv.MouseEvent (new MouseEvent {
  561. // X = 0,
  562. // Y = 4,
  563. // Flags = MouseFlags.Button1Clicked
  564. // }));
  565. // Assert.Equal ("Three", selected);
  566. // Assert.Equal (2, lv.SelectedItem);
  567. // }
  568. }