ApplicationTests.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053
  1. using Xunit.Abstractions;
  2. // Alias Console to MockConsole so we don't accidentally use Console
  3. namespace Terminal.Gui.ApplicationTests;
  4. public class ApplicationTests
  5. {
  6. private readonly ITestOutputHelper _output;
  7. public ApplicationTests (ITestOutputHelper output)
  8. {
  9. _output = output;
  10. ConsoleDriver.RunningUnitTests = true;
  11. #if DEBUG_IDISPOSABLE
  12. Responder.Instances.Clear ();
  13. RunState.Instances.Clear ();
  14. #endif
  15. }
  16. [Fact]
  17. public void Begin_Null_Toplevel_Throws ()
  18. {
  19. // Setup Mock driver
  20. Init ();
  21. // Test null Toplevel
  22. Assert.Throws<ArgumentNullException> (() => Application.Begin (null));
  23. Shutdown ();
  24. Assert.Null (Application.Top);
  25. Assert.Null (Application.MainLoop);
  26. Assert.Null (Application.Driver);
  27. }
  28. [Fact]
  29. [AutoInitShutdown]
  30. public void Begin_Sets_Application_Top_To_Console_Size ()
  31. {
  32. Assert.Equal (new Rectangle (0, 0, 80, 25), Application.Top.Frame);
  33. Application.Begin (Application.Top);
  34. Assert.Equal (new Rectangle (0, 0, 80, 25), Application.Top.Frame);
  35. ((FakeDriver)Application.Driver).SetBufferSize (5, 5);
  36. Assert.Equal (new Rectangle (0, 0, 5, 5), Application.Top.Frame);
  37. }
  38. [Fact]
  39. public void End_Should_Not_Dispose_ApplicationTop_Shutdown_Should ()
  40. {
  41. Init ();
  42. RunState rs = Application.Begin (Application.Top);
  43. Assert.Equal (rs.Toplevel, Application.Top);
  44. Application.End (rs);
  45. #if DEBUG_IDISPOSABLE
  46. Assert.True (rs.WasDisposed);
  47. Assert.True (Application.Top.WasDisposed); // Is true because the rs.Toplevel is the same as Application.Top
  48. #endif
  49. Assert.Null (rs.Toplevel);
  50. var top = Application.Top;
  51. Shutdown ();
  52. #if DEBUG_IDISPOSABLE
  53. Assert.True (top.WasDisposed);
  54. #endif
  55. Assert.Null (Application.Top);
  56. }
  57. [Fact]
  58. public void Init_Begin_End_Cleans_Up ()
  59. {
  60. Init ();
  61. // Begin will cause Run() to be called, which will call Begin(). Thus will block the tests
  62. // if we don't stop
  63. Application.Iteration += (s, a) => { Application.RequestStop (); };
  64. RunState runstate = null;
  65. EventHandler<RunStateEventArgs> NewRunStateFn = (s, e) =>
  66. {
  67. Assert.NotNull (e.State);
  68. runstate = e.State;
  69. };
  70. Application.NotifyNewRunState += NewRunStateFn;
  71. var topLevel = new Toplevel ();
  72. RunState rs = Application.Begin (topLevel);
  73. Assert.NotNull (rs);
  74. Assert.NotNull (runstate);
  75. Assert.Equal (rs, runstate);
  76. Assert.Equal (topLevel, Application.Top);
  77. Assert.Equal (topLevel, Application.Current);
  78. Application.NotifyNewRunState -= NewRunStateFn;
  79. Application.End (runstate);
  80. Assert.Null (Application.Current);
  81. Assert.NotNull (Application.Top);
  82. Assert.NotNull (Application.MainLoop);
  83. Assert.NotNull (Application.Driver);
  84. Shutdown ();
  85. Assert.Null (Application.Top);
  86. Assert.Null (Application.MainLoop);
  87. Assert.Null (Application.Driver);
  88. }
  89. [Theory]
  90. [InlineData (typeof (FakeDriver))]
  91. [InlineData (typeof (NetDriver))]
  92. //[InlineData (typeof (ANSIDriver))]
  93. [InlineData (typeof (WindowsDriver))]
  94. [InlineData (typeof (CursesDriver))]
  95. public void Init_DriverName_Should_Pick_Correct_Driver (Type driverType)
  96. {
  97. var driver = (ConsoleDriver)Activator.CreateInstance (driverType);
  98. Application.Init (driverName: driverType.Name);
  99. Assert.NotNull (Application.Driver);
  100. Assert.Equal (driverType, Application.Driver.GetType ());
  101. Shutdown ();
  102. }
  103. [Fact]
  104. public void Init_Null_Driver_Should_Pick_A_Driver ()
  105. {
  106. Application.Init ();
  107. Assert.NotNull (Application.Driver);
  108. Shutdown ();
  109. }
  110. [Fact]
  111. public void Init_ResetState_Resets_Properties ()
  112. {
  113. ConfigurationManager.ThrowOnJsonErrors = true;
  114. // For all the fields/properties of Application, check that they are reset to their default values
  115. // Set some values
  116. Application.Init ();
  117. Application._initialized = true;
  118. // Reset
  119. Application.ResetState ();
  120. void CheckReset ()
  121. {
  122. // Check that all fields and properties are set to their default values
  123. // Public Properties
  124. Assert.Null (Application.Top);
  125. Assert.Null (Application.Current);
  126. Assert.Null (Application.MouseGrabView);
  127. Assert.Null (Application.WantContinuousButtonPressedView);
  128. // Don't check Application.ForceDriver
  129. // Assert.Empty (Application.ForceDriver);
  130. Assert.False (Application.Force16Colors);
  131. Assert.Null (Application.Driver);
  132. Assert.Null (Application.MainLoop);
  133. Assert.False (Application.EndAfterFirstIteration);
  134. Assert.Equal (Key.Empty, Application.AlternateBackwardKey);
  135. Assert.Equal (Key.Empty, Application.AlternateForwardKey);
  136. Assert.Equal (Key.Empty, Application.QuitKey);
  137. Assert.Null (Application.OverlappedChildren);
  138. Assert.Null (Application.OverlappedTop);
  139. // Internal properties
  140. Assert.False (Application._initialized);
  141. Assert.Equal (Application.GetSupportedCultures (), Application.SupportedCultures);
  142. Assert.False (Application._forceFakeConsole);
  143. Assert.Equal (-1, Application._mainThreadId);
  144. Assert.Empty (Application._topLevels);
  145. Assert.Null (Application._mouseEnteredView);
  146. // Events - Can't check
  147. //Assert.Null (Application.NotifyNewRunState);
  148. //Assert.Null (Application.NotifyNewRunState);
  149. //Assert.Null (Application.Iteration);
  150. //Assert.Null (Application.SizeChanging);
  151. //Assert.Null (Application.GrabbedMouse);
  152. //Assert.Null (Application.UnGrabbingMouse);
  153. //Assert.Null (Application.GrabbedMouse);
  154. //Assert.Null (Application.UnGrabbedMouse);
  155. //Assert.Null (Application.MouseEvent);
  156. //Assert.Null (Application.KeyDown);
  157. //Assert.Null (Application.KeyUp);
  158. }
  159. CheckReset ();
  160. // Set the values that can be set
  161. Application._initialized = true;
  162. Application._forceFakeConsole = true;
  163. Application._mainThreadId = 1;
  164. //Application._topLevels = new List<Toplevel> ();
  165. Application._mouseEnteredView = new View ();
  166. //Application.SupportedCultures = new List<CultureInfo> ();
  167. Application.Force16Colors = true;
  168. //Application.ForceDriver = "driver";
  169. Application.EndAfterFirstIteration = true;
  170. Application.AlternateBackwardKey = Key.A;
  171. Application.AlternateForwardKey = Key.B;
  172. Application.QuitKey = Key.C;
  173. //Application.OverlappedChildren = new List<View> ();
  174. //Application.OverlappedTop =
  175. Application._mouseEnteredView = new View ();
  176. //Application.WantContinuousButtonPressedView = new View ();
  177. Application.ResetState ();
  178. CheckReset ();
  179. ConfigurationManager.ThrowOnJsonErrors = false;
  180. }
  181. [Fact]
  182. public void Init_Shutdown_Cleans_Up ()
  183. {
  184. // Verify initial state is per spec
  185. //Pre_Init_State ();
  186. Application.Init (new FakeDriver ());
  187. // Verify post-Init state is correct
  188. //Post_Init_State ();
  189. Application.Shutdown ();
  190. // Verify state is back to initial
  191. //Pre_Init_State ();
  192. #if DEBUG_IDISPOSABLE
  193. // Validate there are no outstanding Responder-based instances
  194. // after a scenario was selected to run. This proves the main UI Catalog
  195. // 'app' closed cleanly.
  196. Assert.Empty (Responder.Instances);
  197. #endif
  198. }
  199. [Fact]
  200. public void Init_Unbalanced_Throws ()
  201. {
  202. Application.Init (new FakeDriver ());
  203. Toplevel topLevel = null;
  204. Assert.Throws<InvalidOperationException> (
  205. () =>
  206. Application.InternalInit (
  207. new FakeDriver ()
  208. )
  209. );
  210. Shutdown ();
  211. Assert.Null (Application.Top);
  212. Assert.Null (Application.MainLoop);
  213. Assert.Null (Application.Driver);
  214. // Now try the other way
  215. topLevel = null;
  216. Application.InternalInit (new FakeDriver ());
  217. Assert.Throws<InvalidOperationException> (() => Application.Init (new FakeDriver ()));
  218. Shutdown ();
  219. Assert.Null (Application.Top);
  220. Assert.Null (Application.MainLoop);
  221. Assert.Null (Application.Driver);
  222. }
  223. [Fact]
  224. public void InitWithTopLevelFactory_Begin_End_Cleans_Up ()
  225. {
  226. // Begin will cause Run() to be called, which will call Begin(). Thus will block the tests
  227. // if we don't stop
  228. Application.Iteration += (s, a) => { Application.RequestStop (); };
  229. // NOTE: Run<T>, when called after Init has been called behaves differently than
  230. // when called if Init has not been called.
  231. Toplevel topLevel = null;
  232. Application.InternalInit (new FakeDriver ());
  233. RunState runstate = null;
  234. EventHandler<RunStateEventArgs> NewRunStateFn = (s, e) =>
  235. {
  236. Assert.NotNull (e.State);
  237. runstate = e.State;
  238. };
  239. Application.NotifyNewRunState += NewRunStateFn;
  240. RunState rs = Application.Begin (topLevel);
  241. Assert.NotNull (rs);
  242. Assert.NotNull (runstate);
  243. Assert.Equal (rs, runstate);
  244. Assert.Equal (topLevel, Application.Top);
  245. Assert.Equal (topLevel, Application.Current);
  246. Application.NotifyNewRunState -= NewRunStateFn;
  247. Application.End (runstate);
  248. Assert.Null (Application.Current);
  249. Assert.NotNull (Application.Top);
  250. Assert.NotNull (Application.MainLoop);
  251. Assert.NotNull (Application.Driver);
  252. Shutdown ();
  253. Assert.Null (Application.Top);
  254. Assert.Null (Application.MainLoop);
  255. Assert.Null (Application.Driver);
  256. }
  257. [Fact]
  258. [AutoInitShutdown]
  259. public void Internal_Properties_Correct ()
  260. {
  261. Assert.True (Application._initialized);
  262. Assert.NotNull (Application.Top);
  263. RunState rs = Application.Begin (Application.Top);
  264. Assert.Equal (Application.Top, rs.Toplevel);
  265. Assert.Null (Application.MouseGrabView); // public
  266. Assert.Null (Application.WantContinuousButtonPressedView); // public
  267. Assert.False (Application.MoveToOverlappedChild (Application.Top));
  268. }
  269. // Invoke Tests
  270. // TODO: Test with threading scenarios
  271. [Fact]
  272. public void Invoke_Adds_Idle ()
  273. {
  274. Application.Init (new FakeDriver ());
  275. var top = new Toplevel ();
  276. RunState rs = Application.Begin (top);
  277. var firstIteration = false;
  278. var actionCalled = 0;
  279. Application.Invoke (() => { actionCalled++; });
  280. Application.MainLoop.Running = true;
  281. Application.RunIteration (ref rs, ref firstIteration);
  282. Assert.Equal (1, actionCalled);
  283. Application.Shutdown ();
  284. }
  285. [Fact]
  286. [AutoInitShutdown]
  287. public void SetCurrentAsTop_Run_A_Not_Modal_Toplevel_Make_It_The_Current_Application_Top ()
  288. {
  289. var top = Application.Top;
  290. var t1 = new Toplevel ();
  291. var t2 = new Toplevel ();
  292. var t3 = new Toplevel ();
  293. // Don't use Dialog here as it has more layout logic. Use Window instead.
  294. var d = new Dialog ();
  295. var t4 = new Toplevel ();
  296. // t1, t2, t3, d, t4
  297. var iterations = 5;
  298. t1.Ready += (s, e) =>
  299. {
  300. Assert.Equal (t1, Application.Top);
  301. Application.Run (t2);
  302. };
  303. t2.Ready += (s, e) =>
  304. {
  305. Assert.Equal (t2, Application.Top);
  306. Application.Run (t3);
  307. };
  308. t3.Ready += (s, e) =>
  309. {
  310. Assert.Equal (t3, Application.Top);
  311. Application.Run (d);
  312. };
  313. d.Ready += (s, e) =>
  314. {
  315. Assert.Equal (t3, Application.Top);
  316. Application.Run (t4);
  317. };
  318. t4.Ready += (s, e) =>
  319. {
  320. Assert.Equal (t4, Application.Top);
  321. t4.RequestStop ();
  322. d.RequestStop ();
  323. t3.RequestStop ();
  324. t2.RequestStop ();
  325. };
  326. // Now this will close the OverlappedContainer when all OverlappedChildren was closed
  327. t2.Closed += (s, _) => { t1.RequestStop (); };
  328. Application.Iteration += (s, a) =>
  329. {
  330. if (iterations == 5)
  331. {
  332. // The Current still is t4 because Current.Running is false.
  333. Assert.Equal (t4, Application.Current);
  334. Assert.False (Application.Current.Running);
  335. Assert.Equal (t4, Application.Top);
  336. }
  337. else if (iterations == 4)
  338. {
  339. // The Current is d and Current.Running is false.
  340. Assert.Equal (d, Application.Current);
  341. Assert.False (Application.Current.Running);
  342. Assert.Equal (t4, Application.Top);
  343. }
  344. else if (iterations == 3)
  345. {
  346. // The Current is t3 and Current.Running is false.
  347. Assert.Equal (t3, Application.Current);
  348. Assert.False (Application.Current.Running);
  349. Assert.Equal (t3, Application.Top);
  350. }
  351. else if (iterations == 2)
  352. {
  353. // The Current is t2 and Current.Running is false.
  354. Assert.Equal (t2, Application.Current);
  355. Assert.False (Application.Current.Running);
  356. Assert.Equal (t2, Application.Top);
  357. }
  358. else
  359. {
  360. // The Current is t1.
  361. Assert.Equal (t1, Application.Current);
  362. Assert.False (Application.Current.Running);
  363. Assert.Equal (t1, Application.Top);
  364. }
  365. iterations--;
  366. };
  367. Application.Run (t1);
  368. Assert.NotEqual (t1, Application.Top);
  369. Assert.Equal (top, Application.Top);
  370. #if DEBUG_IDISPOSABLE
  371. Assert.True (Application.Top.WasDisposed);
  372. #endif
  373. }
  374. private void Init ()
  375. {
  376. Application.Init (new FakeDriver ());
  377. Assert.NotNull (Application.Driver);
  378. Assert.NotNull (Application.MainLoop);
  379. Assert.NotNull (SynchronizationContext.Current);
  380. }
  381. private void Post_Init_State ()
  382. {
  383. Assert.NotNull (Application.Driver);
  384. Assert.NotNull (Application.Top);
  385. Assert.NotNull (Application.Current);
  386. Assert.NotNull (Application.MainLoop);
  387. // FakeDriver is always 80x25
  388. Assert.Equal (80, Application.Driver.Cols);
  389. Assert.Equal (25, Application.Driver.Rows);
  390. }
  391. private void Pre_Init_State ()
  392. {
  393. Assert.Null (Application.Driver);
  394. Assert.Null (Application.Top);
  395. Assert.Null (Application.Current);
  396. Assert.Null (Application.MainLoop);
  397. }
  398. private void Shutdown () { Application.Shutdown (); }
  399. private class TestToplevel : Toplevel
  400. {
  401. public TestToplevel () { IsOverlappedContainer = false; }
  402. }
  403. #region RunTests
  404. [Fact]
  405. public void Run_T_After_InitWithDriver_with_TopLevel_Throws ()
  406. {
  407. // Setup Mock driver
  408. Init ();
  409. // Run<Toplevel> when already initialized with a Driver will throw (because Toplevel is not derived from TopLevel)
  410. Assert.Throws<ArgumentException> (() => Application.Run<Toplevel> ());
  411. Shutdown ();
  412. Assert.Null (Application.Top);
  413. Assert.Null (Application.MainLoop);
  414. Assert.Null (Application.Driver);
  415. }
  416. [Fact]
  417. public void Run_T_After_InitWithDriver_with_TopLevel_and_Driver_Throws ()
  418. {
  419. // Setup Mock driver
  420. Init ();
  421. // Run<Toplevel> when already initialized with a Driver will throw (because Toplevel is not derivied from TopLevel)
  422. Assert.Throws<ArgumentException> (() => Application.Run<Toplevel> (null, new FakeDriver ()));
  423. Shutdown ();
  424. Assert.Null (Application.Top);
  425. Assert.Null (Application.MainLoop);
  426. Assert.Null (Application.Driver);
  427. }
  428. [Fact]
  429. [TestRespondersDisposed]
  430. public void Run_T_After_Init_Disposes_Application_Top ()
  431. {
  432. Init ();
  433. // Init created a Toplevel and assigned it to Application.Top
  434. var initTop = Application.Top;
  435. Application.Iteration += (s, a) =>
  436. {
  437. Assert.NotEqual(initTop, Application.Top);
  438. #if DEBUG_IDISPOSABLE
  439. Assert.True (initTop.WasDisposed);
  440. #endif
  441. Application.RequestStop ();
  442. };
  443. Application.Run<TestToplevel> ();
  444. #if DEBUG_IDISPOSABLE
  445. Assert.True (initTop.WasDisposed);
  446. #endif
  447. Shutdown ();
  448. Assert.Null (Application.Top);
  449. Assert.Null (Application.MainLoop);
  450. Assert.Null (Application.Driver);
  451. }
  452. [Fact]
  453. [TestRespondersDisposed]
  454. public void Run_T_After_InitWithDriver_with_TestTopLevel_DoesNotThrow ()
  455. {
  456. // Setup Mock driver
  457. Init ();
  458. Application.Iteration += (s, a) => { Application.RequestStop (); };
  459. // Init has been called and we're passing no driver to Run<TestTopLevel>. This is ok.
  460. Application.Run<TestToplevel> ();
  461. Shutdown ();
  462. Assert.Null (Application.Top);
  463. Assert.Null (Application.MainLoop);
  464. Assert.Null (Application.Driver);
  465. }
  466. [Fact]
  467. [TestRespondersDisposed]
  468. public void Run_T_After_InitNullDriver_with_TestTopLevel_DoesNotThrow ()
  469. {
  470. Application.ForceDriver = "FakeDriver";
  471. Application.Init ();
  472. Assert.Equal (typeof (FakeDriver), Application.Driver.GetType ());
  473. Application.Iteration += (s, a) => { Application.RequestStop (); };
  474. // Init has been called, selecting FakeDriver; we're passing no driver to Run<TestTopLevel>. Should be fine.
  475. Application.Run<TestToplevel> ();
  476. Shutdown ();
  477. Assert.Null (Application.Top);
  478. Assert.Null (Application.MainLoop);
  479. Assert.Null (Application.Driver);
  480. }
  481. [Fact]
  482. [TestRespondersDisposed]
  483. public void Run_T_Init_Driver_Cleared_with_TestTopLevel_Throws ()
  484. {
  485. Init ();
  486. Application.Driver = null;
  487. Application.Iteration += (s, a) => { Application.RequestStop (); };
  488. // Init has been called, but Driver has been set to null. Bad.
  489. Assert.Throws<InvalidOperationException> (() => Application.Run<TestToplevel> ());
  490. Shutdown ();
  491. Assert.Null (Application.Top);
  492. Assert.Null (Application.MainLoop);
  493. Assert.Null (Application.Driver);
  494. }
  495. [Fact]
  496. [TestRespondersDisposed]
  497. public void Run_T_NoInit_DoesNotThrow ()
  498. {
  499. Application.ForceDriver = "FakeDriver";
  500. Application.Iteration += (s, a) => { Application.RequestStop (); };
  501. Application.Run<TestToplevel> ();
  502. Assert.Equal (typeof (FakeDriver), Application.Driver.GetType ());
  503. Shutdown ();
  504. Assert.Null (Application.Top);
  505. Assert.Null (Application.MainLoop);
  506. Assert.Null (Application.Driver);
  507. }
  508. [Fact]
  509. [TestRespondersDisposed]
  510. public void Run_T_NoInit_WithDriver_DoesNotThrow ()
  511. {
  512. Application.Iteration += (s, a) => { Application.RequestStop (); };
  513. // Init has NOT been called and we're passing a valid driver to Run<TestTopLevel>. This is ok.
  514. Application.Run<TestToplevel> (null, new FakeDriver ());
  515. Shutdown ();
  516. Assert.Null (Application.Top);
  517. Assert.Null (Application.MainLoop);
  518. Assert.Null (Application.Driver);
  519. }
  520. [Fact]
  521. [TestRespondersDisposed]
  522. public void Run_RequestStop_Stops ()
  523. {
  524. // Setup Mock driver
  525. Init ();
  526. var top = new Toplevel ();
  527. RunState rs = Application.Begin (top);
  528. Assert.NotNull (rs);
  529. Assert.Equal (top, Application.Current);
  530. Application.Iteration += (s, a) => { Application.RequestStop (); };
  531. Application.Run (top);
  532. Application.Shutdown ();
  533. Assert.Null (Application.Current);
  534. Assert.Null (Application.Top);
  535. Assert.Null (Application.MainLoop);
  536. Assert.Null (Application.Driver);
  537. }
  538. [Fact]
  539. [TestRespondersDisposed]
  540. public void Run_RunningFalse_Stops ()
  541. {
  542. // Setup Mock driver
  543. Init ();
  544. var top = new Toplevel ();
  545. RunState rs = Application.Begin (top);
  546. Assert.NotNull (rs);
  547. Assert.Equal (top, Application.Current);
  548. Application.Iteration += (s, a) => { top.Running = false; };
  549. Application.Run (top);
  550. Application.Shutdown ();
  551. Assert.Null (Application.Current);
  552. Assert.Null (Application.Top);
  553. Assert.Null (Application.MainLoop);
  554. Assert.Null (Application.Driver);
  555. }
  556. [Fact]
  557. [TestRespondersDisposed]
  558. public void Run_Loaded_Ready_Unlodaded_Events ()
  559. {
  560. Init ();
  561. Toplevel top = Application.Top;
  562. var count = 0;
  563. top.Loaded += (s, e) => count++;
  564. top.Ready += (s, e) => count++;
  565. top.Unloaded += (s, e) => count++;
  566. Application.Iteration += (s, a) => Application.RequestStop ();
  567. Application.Run ();
  568. Application.Shutdown ();
  569. Assert.Equal (3, count);
  570. }
  571. // TODO: All Toplevel layout tests should be moved to ToplevelTests.cs
  572. [Fact]
  573. public void Run_Toplevel_With_Modal_View_Does_Not_Refresh_If_Not_Dirty ()
  574. {
  575. Init ();
  576. var count = 0;
  577. // Don't use Dialog here as it has more layout logic. Use Window instead.
  578. Dialog d = null;
  579. Toplevel top = Application.Top;
  580. top.DrawContent += (s, a) => count++;
  581. int iteration = -1;
  582. Application.Iteration += (s, a) =>
  583. {
  584. iteration++;
  585. if (iteration == 0)
  586. {
  587. // TODO: Don't use Dialog here as it has more layout logic. Use Window instead.
  588. d = new Dialog ();
  589. d.DrawContent += (s, a) => count++;
  590. Application.Run (d);
  591. }
  592. else if (iteration < 3)
  593. {
  594. Application.OnMouseEvent (
  595. new MouseEventEventArgs (
  596. new MouseEvent
  597. { X = 0, Y = 0, Flags = MouseFlags.ReportMousePosition }
  598. )
  599. );
  600. Assert.False (top.NeedsDisplay);
  601. Assert.False (top.SubViewNeedsDisplay);
  602. Assert.False (top.LayoutNeeded);
  603. Assert.False (d.NeedsDisplay);
  604. Assert.False (d.SubViewNeedsDisplay);
  605. Assert.False (d.LayoutNeeded);
  606. }
  607. else
  608. {
  609. Application.RequestStop ();
  610. }
  611. };
  612. Application.Run ();
  613. Application.Shutdown ();
  614. // 1 - First top load, 1 - Dialog load, 1 - Dialog unload, Total - 3.
  615. Assert.Equal (3, count);
  616. }
  617. // TODO: All Toplevel layout tests should be moved to ToplevelTests.cs
  618. [Fact]
  619. public void Run_A_Modal_Toplevel_Refresh_Background_On_Moving ()
  620. {
  621. Init ();
  622. // Don't use Dialog here as it has more layout logic. Use Window instead.
  623. var w = new Window { Width = 5, Height = 5 };
  624. ((FakeDriver)Application.Driver).SetBufferSize (10, 10);
  625. RunState rs = Application.Begin (w);
  626. TestHelpers.AssertDriverContentsWithFrameAre (
  627. @"
  628. ┌───┐
  629. │ │
  630. │ │
  631. │ │
  632. └───┘",
  633. _output
  634. );
  635. Attribute [] attributes =
  636. {
  637. // 0
  638. new (ColorName.White, ColorName.Black),
  639. // 1
  640. Colors.ColorSchemes ["Base"].Normal
  641. };
  642. TestHelpers.AssertDriverAttributesAre (
  643. @"
  644. 1111100000
  645. 1111100000
  646. 1111100000
  647. 1111100000
  648. 1111100000
  649. ",
  650. null,
  651. attributes
  652. );
  653. // TODO: In PR #2920 this breaks because the mouse is not grabbed anymore.
  654. // TODO: Move the mouse grap/drag mode from Toplevel to Border.
  655. Application.OnMouseEvent (
  656. new MouseEventEventArgs (
  657. new MouseEvent { X = 0, Y = 0, Flags = MouseFlags.Button1Pressed }
  658. )
  659. );
  660. Assert.Equal (w.Border, Application.MouseGrabView);
  661. // Move down and to the right.
  662. Application.OnMouseEvent (
  663. new MouseEventEventArgs (
  664. new MouseEvent
  665. {
  666. X = 1,
  667. Y = 1,
  668. Flags = MouseFlags.Button1Pressed
  669. | MouseFlags.ReportMousePosition
  670. }
  671. )
  672. );
  673. Application.Refresh ();
  674. TestHelpers.AssertDriverContentsWithFrameAre (
  675. @"
  676. ┌───┐
  677. │ │
  678. │ │
  679. │ │
  680. └───┘",
  681. _output
  682. );
  683. attributes = new []
  684. {
  685. // 0
  686. new (ColorName.White, ColorName.Black),
  687. // 1
  688. Colors.ColorSchemes ["Base"].Normal
  689. };
  690. TestHelpers.AssertDriverAttributesAre (
  691. @"
  692. 0000000000
  693. 0111110000
  694. 0111110000
  695. 0111110000
  696. 0111110000
  697. 0111110000
  698. ",
  699. null,
  700. attributes
  701. );
  702. Application.End (rs);
  703. w.Dispose ();
  704. Application.Shutdown ();
  705. }
  706. [Fact]
  707. public void End_Does_Not_Dispose ()
  708. {
  709. Init ();
  710. var top = new Toplevel ();
  711. Window w = new ();
  712. w.Ready += (s, e) => Application.RequestStop (); // Causes `End` to be called
  713. Application.Run(w);
  714. #if DEBUG_IDISPOSABLE
  715. Assert.False (w.WasDisposed);
  716. #endif
  717. Assert.NotNull (w);
  718. Assert.Equal (string.Empty, w.Title); // Valid - w has not been disposed. The user may want to run it again
  719. Assert.NotNull (Application.Top);
  720. Assert.Equal(w, Application.Top);
  721. Assert.NotEqual(top, Application.Top);
  722. Assert.Null (Application.Current);
  723. Application.Run(w); // Valid - w has not been disposed.
  724. #if DEBUG_IDISPOSABLE
  725. Assert.False (w.WasDisposed);
  726. var exception = Record.Exception (() => Application.Shutdown()); // Invalid - w has not been disposed.
  727. Assert.NotNull (exception);
  728. w.Dispose ();
  729. Assert.True (w.WasDisposed);
  730. exception = Record.Exception (() => Application.Run (w)); // Invalid - w has been disposed. Run it in debug mode will throw, otherwise the user may want to run it again
  731. Assert.NotNull (exception);
  732. exception = Record.Exception (() => Assert.Equal (string.Empty, w.Title)); // Invalid - w has been disposed and cannot be accessed
  733. Assert.NotNull (exception);
  734. exception = Record.Exception (() => w.Title = "NewTitle"); // Invalid - w has been disposed and cannot be accessed
  735. Assert.NotNull (exception);
  736. #endif
  737. Application.Shutdown ();
  738. Assert.NotNull (w);
  739. Assert.Null (Application.Current);
  740. Assert.NotNull (top);
  741. Assert.Null (Application.Top);
  742. }
  743. [Fact]
  744. public void Run_Creates_Top_Without_Init ()
  745. {
  746. var driver = new FakeDriver ();
  747. Assert.Null (Application.Top);
  748. Application.Iteration += (s, e) =>
  749. {
  750. Assert.NotNull (Application.Top);
  751. Application.RequestStop ();
  752. };
  753. Application.Run (null, driver);
  754. #if DEBUG_IDISPOSABLE
  755. Assert.False (Application.Top.WasDisposed);
  756. var exception = Record.Exception (() => Application.Shutdown ());
  757. Assert.NotNull (exception);
  758. Assert.False (Application.Top.WasDisposed);
  759. // It's up to caller to dispose it
  760. Application.Top.Dispose ();
  761. Assert.True (Application.Top.WasDisposed);
  762. #endif
  763. Assert.NotNull (Application.Top);
  764. Application.Shutdown ();
  765. Assert.Null (Application.Top);
  766. }
  767. [Fact]
  768. public void Run_T_Creates_Top_Without_Init ()
  769. {
  770. var driver = new FakeDriver ();
  771. Assert.Null (Application.Top);
  772. Application.Iteration += (s, e) =>
  773. {
  774. Assert.NotNull (Application.Top);
  775. Application.RequestStop ();
  776. };
  777. Application.Run<Toplevel> (null, driver);
  778. #if DEBUG_IDISPOSABLE
  779. Assert.False (Application.Top.WasDisposed);
  780. var exception = Record.Exception (() => Application.Shutdown ());
  781. Assert.NotNull (exception);
  782. Assert.False (Application.Top.WasDisposed);
  783. // It's up to caller to dispose it
  784. Application.Top.Dispose ();
  785. Assert.True (Application.Top.WasDisposed);
  786. #endif
  787. Assert.NotNull (Application.Top);
  788. Application.Shutdown ();
  789. Assert.Null (Application.Top);
  790. }
  791. [Fact]
  792. public void Run_t_Creates_Top_Without_Init ()
  793. {
  794. var driver = new FakeDriver ();
  795. Assert.Null (Application.Top);
  796. Application.Iteration += (s, e) =>
  797. {
  798. Assert.NotNull (Application.Top);
  799. Application.RequestStop ();
  800. };
  801. Application.Run (new (), null, driver);
  802. #if DEBUG_IDISPOSABLE
  803. Assert.False (Application.Top.WasDisposed);
  804. var exception = Record.Exception (() => Application.Shutdown ());
  805. Assert.NotNull (exception);
  806. Assert.False (Application.Top.WasDisposed);
  807. // It's up to caller to dispose it
  808. Application.Top.Dispose ();
  809. Assert.True (Application.Top.WasDisposed);
  810. #endif
  811. Assert.NotNull (Application.Top);
  812. Application.Shutdown ();
  813. Assert.Null (Application.Top);
  814. }
  815. // TODO: Add tests for Run that test errorHandler
  816. #endregion
  817. #region ShutdownTests
  818. [Fact]
  819. public async void Shutdown_Allows_Async ()
  820. {
  821. var isCompletedSuccessfully = false;
  822. async Task TaskWithAsyncContinuation ()
  823. {
  824. await Task.Yield ();
  825. await Task.Yield ();
  826. isCompletedSuccessfully = true;
  827. }
  828. Init ();
  829. Application.Shutdown ();
  830. Assert.False (isCompletedSuccessfully);
  831. await TaskWithAsyncContinuation ();
  832. Thread.Sleep (100);
  833. Assert.True (isCompletedSuccessfully);
  834. }
  835. [Fact]
  836. public void Shutdown_Resets_SyncContext ()
  837. {
  838. Init ();
  839. Application.Shutdown ();
  840. Assert.Null (SynchronizationContext.Current);
  841. }
  842. #endregion
  843. }