ButtonTests.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547
  1. using UnitTests;
  2. using Xunit.Abstractions;
  3. namespace Terminal.Gui.ViewsTests;
  4. public class ButtonTests (ITestOutputHelper output)
  5. {
  6. [Fact]
  7. [SetupFakeDriver]
  8. public void Constructors_Defaults ()
  9. {
  10. // Override CM
  11. Button.DefaultShadow = ShadowStyle.None;
  12. var btn = new Button ();
  13. Assert.Equal (string.Empty, btn.Text);
  14. btn.BeginInit ();
  15. btn.EndInit ();
  16. btn.SetRelativeLayout (new (100, 100));
  17. Assert.Equal ($"{Glyphs.LeftBracket} {Glyphs.RightBracket}", btn.TextFormatter.Text);
  18. Assert.False (btn.IsDefault);
  19. Assert.Equal (Alignment.Center, btn.TextAlignment);
  20. Assert.Equal ('_', btn.HotKeySpecifier.Value);
  21. Assert.True (btn.CanFocus);
  22. Assert.Equal (new (0, 0, 4, 1), btn.Viewport);
  23. Assert.Equal (new (0, 0, 4, 1), btn.Frame);
  24. Assert.Equal ($"{Glyphs.LeftBracket} {Glyphs.RightBracket}", btn.TextFormatter.Text);
  25. Assert.False (btn.IsDefault);
  26. Assert.Equal (Alignment.Center, btn.TextAlignment);
  27. Assert.Equal ('_', btn.HotKeySpecifier.Value);
  28. Assert.True (btn.CanFocus);
  29. Assert.Equal (new (0, 0, 4, 1), btn.Viewport);
  30. Assert.Equal (new (0, 0, 4, 1), btn.Frame);
  31. Assert.Equal (string.Empty, btn.Title);
  32. Assert.Equal (KeyCode.Null, btn.HotKey);
  33. btn.Draw ();
  34. var expected = @$"
  35. {Glyphs.LeftBracket} {Glyphs.RightBracket}
  36. ";
  37. DriverAssert.AssertDriverContentsWithFrameAre (expected, output);
  38. btn.Dispose ();
  39. btn = new () { Text = "_Test", IsDefault = true };
  40. btn.Layout ();
  41. Assert.Equal (new (10, 1), btn.TextFormatter.ConstrainToSize);
  42. btn.BeginInit ();
  43. btn.EndInit ();
  44. Assert.Equal ('_', btn.HotKeySpecifier.Value);
  45. Assert.Equal (Key.T, btn.HotKey);
  46. Assert.Equal ("_Test", btn.Text);
  47. Assert.Equal (
  48. $"{Glyphs.LeftBracket}{Glyphs.LeftDefaultIndicator} Test {Glyphs.RightDefaultIndicator}{Glyphs.RightBracket}",
  49. btn.TextFormatter.Format ()
  50. );
  51. Assert.True (btn.IsDefault);
  52. Assert.Equal (Alignment.Center, btn.TextAlignment);
  53. Assert.True (btn.CanFocus);
  54. btn.SetRelativeLayout (new (100, 100));
  55. // 0123456789012345678901234567890123456789
  56. // [* Test *]
  57. Assert.Equal ('_', btn.HotKeySpecifier.Value);
  58. Assert.Equal (10, btn.TextFormatter.Format ().Length);
  59. Assert.Equal (new (10, 1), btn.TextFormatter.ConstrainToSize);
  60. Assert.Equal (new (10, 1), btn.GetContentSize ());
  61. Assert.Equal (new (0, 0, 10, 1), btn.Viewport);
  62. Assert.Equal (new (0, 0, 10, 1), btn.Frame);
  63. Assert.Equal (KeyCode.T, btn.HotKey);
  64. btn.Dispose ();
  65. btn = new () { X = 1, Y = 2, Text = "_abc", IsDefault = true };
  66. btn.BeginInit ();
  67. btn.EndInit ();
  68. Assert.Equal ("_abc", btn.Text);
  69. Assert.Equal (Key.A, btn.HotKey);
  70. Assert.Equal (
  71. $"{Glyphs.LeftBracket}{Glyphs.LeftDefaultIndicator} abc {Glyphs.RightDefaultIndicator}{Glyphs.RightBracket}",
  72. btn.TextFormatter.Format ()
  73. );
  74. Assert.True (btn.IsDefault);
  75. Assert.Equal (Alignment.Center, btn.TextAlignment);
  76. Assert.Equal ('_', btn.HotKeySpecifier.Value);
  77. Assert.True (btn.CanFocus);
  78. Application.Driver?.ClearContents ();
  79. btn.Draw ();
  80. expected = @$"
  81. {Glyphs.LeftBracket}{Glyphs.LeftDefaultIndicator} abc {Glyphs.RightDefaultIndicator}{Glyphs.RightBracket}
  82. ";
  83. DriverAssert.AssertDriverContentsWithFrameAre (expected, output);
  84. Assert.Equal (new (0, 0, 9, 1), btn.Viewport);
  85. Assert.Equal (new (1, 2, 9, 1), btn.Frame);
  86. btn.Dispose ();
  87. }
  88. [Fact]
  89. public void HotKeyChange_Works ()
  90. {
  91. var clicked = false;
  92. var btn = new Button { Text = "_Test" };
  93. btn.Accepting += (s, e) => clicked = true;
  94. Assert.Equal (KeyCode.T, btn.HotKey);
  95. Assert.False (btn.NewKeyDownEvent (Key.T)); // Button processes, but does not handle
  96. Assert.True (clicked);
  97. clicked = false;
  98. Assert.False (btn.NewKeyDownEvent (Key.T.WithAlt)); // Button processes, but does not handle
  99. Assert.True (clicked);
  100. clicked = false;
  101. btn.HotKey = KeyCode.E;
  102. Assert.False (btn.NewKeyDownEvent (Key.E.WithAlt)); // Button processes, but does not handle
  103. Assert.True (clicked);
  104. }
  105. [Theory]
  106. [InlineData (false, 0)]
  107. [InlineData (true, 1)]
  108. public void Space_Fires_Accept (bool focused, int expected)
  109. {
  110. var superView = new View
  111. {
  112. CanFocus = true
  113. };
  114. Button button = new ();
  115. button.CanFocus = focused;
  116. var acceptInvoked = 0;
  117. button.Accepting += (s, e) => acceptInvoked++;
  118. superView.Add (button);
  119. button.SetFocus ();
  120. Assert.Equal (focused, button.HasFocus);
  121. superView.NewKeyDownEvent (Key.Space);
  122. Assert.Equal (expected, acceptInvoked);
  123. superView.Dispose ();
  124. }
  125. [Theory]
  126. [InlineData (false, 0)]
  127. [InlineData (true, 1)]
  128. public void Enter_Fires_Accept (bool focused, int expected)
  129. {
  130. var superView = new View
  131. {
  132. CanFocus = true
  133. };
  134. Button button = new ();
  135. button.CanFocus = focused;
  136. var acceptInvoked = 0;
  137. button.Accepting += (s, e) => acceptInvoked++;
  138. superView.Add (button);
  139. button.SetFocus ();
  140. Assert.Equal (focused, button.HasFocus);
  141. superView.NewKeyDownEvent (Key.Enter);
  142. Assert.Equal (expected, acceptInvoked);
  143. superView.Dispose ();
  144. }
  145. [Theory]
  146. [InlineData (false, 1)]
  147. [InlineData (true, 1)]
  148. public void HotKey_Fires_Accept (bool focused, int expected)
  149. {
  150. var superView = new View
  151. {
  152. CanFocus = true
  153. };
  154. Button button = new ()
  155. {
  156. HotKey = Key.A
  157. };
  158. button.CanFocus = focused;
  159. var acceptInvoked = 0;
  160. button.Accepting += (s, e) => acceptInvoked++;
  161. superView.Add (button);
  162. button.SetFocus ();
  163. Assert.Equal (focused, button.HasFocus);
  164. superView.NewKeyDownEvent (Key.A);
  165. Assert.Equal (expected, acceptInvoked);
  166. superView.Dispose ();
  167. }
  168. /// <summary>
  169. /// This test demonstrates how to change the activation key for Button as described in the README.md keyboard
  170. /// handling section
  171. /// </summary>
  172. [Fact]
  173. [AutoInitShutdown]
  174. public void KeyBindingExample ()
  175. {
  176. var pressed = 0;
  177. var btn = new Button { Text = "Press Me" };
  178. btn.Accepting += (s, e) => pressed++;
  179. // The Button class supports the Default and Accept command
  180. Assert.Contains (Command.HotKey, btn.GetSupportedCommands ());
  181. Assert.Contains (Command.Accept, btn.GetSupportedCommands ());
  182. var top = new Toplevel ();
  183. top.Add (btn);
  184. Application.Begin (top);
  185. Assert.True (btn.HasFocus);
  186. // default keybinding is Space which results in Command.Accept (when focused)
  187. Application.RaiseKeyDownEvent (new ((KeyCode)' '));
  188. Assert.Equal (1, pressed);
  189. // remove the default keybinding (Space)
  190. btn.KeyBindings.Clear (Command.HotKey);
  191. btn.KeyBindings.Clear (Command.Accept);
  192. // After clearing the default keystroke the Space button no longer does anything for the Button
  193. Application.RaiseKeyDownEvent (new ((KeyCode)' '));
  194. Assert.Equal (1, pressed);
  195. // Set a new binding of b for the click (Accept) event
  196. btn.KeyBindings.Add (Key.B, Command.HotKey); // b will now trigger the Accept command (when focused or not)
  197. // now pressing B should call the button click event
  198. Application.RaiseKeyDownEvent (Key.B);
  199. Assert.Equal (2, pressed);
  200. // now pressing Shift-B should NOT call the button click event
  201. Application.RaiseKeyDownEvent (Key.B.WithShift);
  202. Assert.Equal (2, pressed);
  203. // now pressing Alt-B should NOT call the button click event
  204. Application.RaiseKeyDownEvent (Key.B.WithAlt);
  205. Assert.Equal (2, pressed);
  206. // now pressing Shift-Alt-B should NOT call the button click event
  207. Application.RaiseKeyDownEvent (Key.B.WithAlt.WithShift);
  208. Assert.Equal (2, pressed);
  209. top.Dispose ();
  210. }
  211. [Fact]
  212. [AutoInitShutdown]
  213. public void KeyBindings_Command ()
  214. {
  215. var clicked = false;
  216. var btn = new Button { Text = "_Test" };
  217. btn.Accepting += (s, e) => clicked = true;
  218. var top = new Toplevel ();
  219. top.Add (btn);
  220. Application.Begin (top);
  221. // Hot key. Both alone and with alt
  222. Assert.Equal (KeyCode.T, btn.HotKey);
  223. Assert.False (btn.NewKeyDownEvent (Key.T)); // Button processes, but does not handle
  224. Assert.True (clicked);
  225. clicked = false;
  226. Assert.False (btn.NewKeyDownEvent (Key.T.WithAlt));
  227. Assert.True (clicked);
  228. clicked = false;
  229. Assert.False (btn.NewKeyDownEvent (btn.HotKey));
  230. Assert.True (clicked);
  231. clicked = false;
  232. Assert.False (btn.NewKeyDownEvent (btn.HotKey));
  233. Assert.True (clicked);
  234. clicked = false;
  235. // IsDefault = false
  236. // Space and Enter should work
  237. Assert.False (btn.IsDefault);
  238. Assert.False (btn.NewKeyDownEvent (Key.Enter));
  239. Assert.True (clicked);
  240. clicked = false;
  241. // IsDefault = true
  242. // Space and Enter should work
  243. btn.IsDefault = true;
  244. Assert.False (btn.NewKeyDownEvent (Key.Enter));
  245. Assert.True (clicked);
  246. clicked = false;
  247. // Toplevel does not handle Enter, so it should get passed on to button
  248. Assert.False (Application.Top.NewKeyDownEvent (Key.Enter));
  249. Assert.True (clicked);
  250. clicked = false;
  251. // Direct
  252. Assert.False (btn.NewKeyDownEvent (Key.Enter));
  253. Assert.True (clicked);
  254. clicked = false;
  255. Assert.False (btn.NewKeyDownEvent (Key.Space));
  256. Assert.True (clicked);
  257. clicked = false;
  258. Assert.False (btn.NewKeyDownEvent (new ((KeyCode)'T')));
  259. Assert.True (clicked);
  260. clicked = false;
  261. // Change hotkey:
  262. btn.Text = "Te_st";
  263. Assert.False (btn.NewKeyDownEvent (btn.HotKey));
  264. Assert.True (clicked);
  265. clicked = false;
  266. top.Dispose ();
  267. }
  268. [Fact]
  269. public void HotKey_Command_Accepts ()
  270. {
  271. var button = new Button ();
  272. var accepted = false;
  273. button.Accepting += ButtonOnAccept;
  274. button.InvokeCommand (Command.HotKey);
  275. Assert.True (accepted);
  276. button.Dispose ();
  277. return;
  278. void ButtonOnAccept (object sender, CommandEventArgs e) { accepted = true; }
  279. }
  280. [Fact]
  281. public void Accept_Cancel_Event_OnAccept_Returns_True ()
  282. {
  283. var button = new Button ();
  284. var acceptInvoked = false;
  285. button.Accepting += ButtonAccept;
  286. bool? ret = button.InvokeCommand (Command.Accept);
  287. Assert.True (ret);
  288. Assert.True (acceptInvoked);
  289. button.Dispose ();
  290. return;
  291. void ButtonAccept (object sender, CommandEventArgs e)
  292. {
  293. acceptInvoked = true;
  294. e.Handled = true;
  295. }
  296. }
  297. [Fact]
  298. public void Setting_Empty_Text_Sets_HoKey_To_KeyNull ()
  299. {
  300. var super = new View ();
  301. var btn = new Button { Text = "_Test" };
  302. super.Add (btn);
  303. super.BeginInit ();
  304. super.EndInit ();
  305. Assert.Equal ("_Test", btn.Text);
  306. Assert.Equal (KeyCode.T, btn.HotKey);
  307. btn.Text = string.Empty;
  308. Assert.Equal ("", btn.Text);
  309. Assert.Equal (KeyCode.Null, btn.HotKey);
  310. btn.Text = string.Empty;
  311. Assert.Equal ("", btn.Text);
  312. Assert.Equal (KeyCode.Null, btn.HotKey);
  313. btn.Text = "Te_st";
  314. Assert.Equal ("Te_st", btn.Text);
  315. Assert.Equal (KeyCode.S, btn.HotKey);
  316. super.Dispose ();
  317. }
  318. [Fact]
  319. public void TestAssignTextToButton ()
  320. {
  321. View b = new Button { Text = "heya" };
  322. Assert.Equal ("heya", b.Text);
  323. Assert.Contains ("heya", b.TextFormatter.Text);
  324. b.Text = "heyb";
  325. Assert.Equal ("heyb", b.Text);
  326. Assert.Contains ("heyb", b.TextFormatter.Text);
  327. // with cast
  328. Assert.Equal ("heyb", ((Button)b).Text);
  329. b.Dispose ();
  330. }
  331. [Fact]
  332. [AutoInitShutdown]
  333. public void Update_Parameterless_Only_On_Or_After_Initialize ()
  334. {
  335. Button.DefaultShadow = ShadowStyle.None;
  336. var btn = new Button { X = Pos.Center (), Y = Pos.Center (), Text = "Say Hello 你" };
  337. var win = new Window { Width = Dim.Fill (), Height = Dim.Fill () };
  338. win.Add (btn);
  339. var top = new Toplevel ();
  340. top.Add (win);
  341. Assert.False (btn.IsInitialized);
  342. Application.Begin (top);
  343. AutoInitShutdownAttribute.FakeResize(new Size(30, 5));
  344. Assert.True (btn.IsInitialized);
  345. Assert.Equal ("Say Hello 你", btn.Text);
  346. Assert.Equal ($"{Glyphs.LeftBracket} {btn.Text} {Glyphs.RightBracket}", btn.TextFormatter.Text);
  347. Assert.Equal (new (0, 0, 16, 1), btn.Viewport);
  348. var btnTxt = $"{Glyphs.LeftBracket} {btn.Text} {Glyphs.RightBracket}";
  349. var expected = @$"
  350. ┌────────────────────────────┐
  351. │ │
  352. │ {btnTxt} │
  353. │ │
  354. └────────────────────────────┘
  355. ";
  356. Rectangle pos = DriverAssert.AssertDriverContentsWithFrameAre (expected, output);
  357. Assert.Equal (new (0, 0, 30, 5), pos);
  358. top.Dispose ();
  359. }
  360. [Theory]
  361. [InlineData (MouseFlags.Button1Pressed, MouseFlags.Button1Released, MouseFlags.Button1Clicked)]
  362. [InlineData (MouseFlags.Button2Pressed, MouseFlags.Button2Released, MouseFlags.Button2Clicked)]
  363. [InlineData (MouseFlags.Button3Pressed, MouseFlags.Button3Released, MouseFlags.Button3Clicked)]
  364. [InlineData (MouseFlags.Button4Pressed, MouseFlags.Button4Released, MouseFlags.Button4Clicked)]
  365. public void WantContinuousButtonPressed_True_ButtonClick_Accepts (MouseFlags pressed, MouseFlags released, MouseFlags clicked)
  366. {
  367. var me = new MouseEventArgs ();
  368. var button = new Button
  369. {
  370. Width = 1,
  371. Height = 1,
  372. WantContinuousButtonPressed = true
  373. };
  374. var selectingCount = 0;
  375. button.Selecting += (s, e) => selectingCount++;
  376. var acceptedCount = 0;
  377. button.Accepting += (s, e) =>
  378. {
  379. acceptedCount++;
  380. e.Handled = true;
  381. };
  382. me = new ();
  383. me.Flags = pressed;
  384. button.NewMouseEvent (me);
  385. Assert.Equal (0, selectingCount);
  386. Assert.Equal (0, acceptedCount);
  387. me = new ();
  388. me.Flags = released;
  389. button.NewMouseEvent (me);
  390. Assert.Equal (0, selectingCount);
  391. Assert.Equal (0, acceptedCount);
  392. me = new ();
  393. me.Flags = clicked;
  394. button.NewMouseEvent (me);
  395. Assert.Equal (1, selectingCount);
  396. Assert.Equal (1, acceptedCount);
  397. button.Dispose ();
  398. }
  399. [Theory]
  400. [InlineData (MouseFlags.Button1Pressed, MouseFlags.Button1Released)]
  401. [InlineData (MouseFlags.Button2Pressed, MouseFlags.Button2Released)]
  402. [InlineData (MouseFlags.Button3Pressed, MouseFlags.Button3Released)]
  403. [InlineData (MouseFlags.Button4Pressed, MouseFlags.Button4Released)]
  404. public void WantContinuousButtonPressed_True_ButtonPressRelease_Does_Not_Raise_Selected_Or_Accepted (MouseFlags pressed, MouseFlags released)
  405. {
  406. var me = new MouseEventArgs ();
  407. var button = new Button
  408. {
  409. Width = 1,
  410. Height = 1,
  411. WantContinuousButtonPressed = true
  412. };
  413. var acceptedCount = 0;
  414. button.Accepting += (s, e) =>
  415. {
  416. acceptedCount++;
  417. e.Handled = true;
  418. };
  419. var selectingCount = 0;
  420. button.Selecting += (s, e) =>
  421. {
  422. selectingCount++;
  423. e.Handled = true;
  424. };
  425. me.Flags = pressed;
  426. button.NewMouseEvent (me);
  427. Assert.Equal (0, acceptedCount);
  428. Assert.Equal (0, selectingCount);
  429. me.Flags = released;
  430. button.NewMouseEvent (me);
  431. Assert.Equal (0, acceptedCount);
  432. Assert.Equal (0, selectingCount);
  433. button.Dispose ();
  434. }
  435. }