ApplicationTests.cs 34 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039
  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. Application.Shutdown ();
  704. }
  705. [Fact]
  706. public void End_Disposing_Correctly ()
  707. {
  708. Init ();
  709. var top = Application.Top;
  710. Window w = new ();
  711. w.Ready += (s, e) => Application.RequestStop (); // Causes `End` to be called
  712. Application.Run(w);
  713. #if DEBUG_IDISPOSABLE
  714. Assert.True (w.WasDisposed);
  715. #endif
  716. Assert.NotNull (w);
  717. Assert.Equal (string.Empty, w.Title); // Invalid - w has been disposed -> Valid - w isn't Application.Top but the original created by Init
  718. Assert.NotNull (Application.Top);
  719. Assert.NotEqual(w, Application.Top);
  720. Assert.Equal(top, Application.Top);
  721. Assert.Null (Application.Current);
  722. var 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
  723. Assert.NotNull (exception);
  724. Application.Shutdown ();
  725. Assert.NotNull (w);
  726. Assert.Equal (string.Empty, w.Title); // Invalid - w has been disposed -> Valid - w isn't Application.Top but the original created by Init
  727. Assert.Null (Application.Current);
  728. Assert.NotNull (top);
  729. Assert.Null (Application.Top);
  730. }
  731. [Fact]
  732. public void Run_Creates_Top_Without_Init ()
  733. {
  734. var driver = new FakeDriver ();
  735. Assert.Null (Application.Top);
  736. Application.Iteration += (s, e) =>
  737. {
  738. Assert.NotNull (Application.Top);
  739. Application.RequestStop ();
  740. };
  741. Application.Run (null, driver);
  742. #if DEBUG_IDISPOSABLE
  743. Assert.False (Application.Top.WasDisposed);
  744. var exception = Record.Exception (() => Application.Shutdown ());
  745. Assert.NotNull (exception);
  746. Assert.False (Application.Top.WasDisposed);
  747. // It's up to caller to dispose it
  748. Application.Top.Dispose ();
  749. Assert.True (Application.Top.WasDisposed);
  750. #endif
  751. Assert.NotNull (Application.Top);
  752. Application.Shutdown ();
  753. Assert.Null (Application.Top);
  754. }
  755. [Fact]
  756. public void Run_T_Creates_Top_Without_Init ()
  757. {
  758. var driver = new FakeDriver ();
  759. Assert.Null (Application.Top);
  760. Application.Iteration += (s, e) =>
  761. {
  762. Assert.NotNull (Application.Top);
  763. Application.RequestStop ();
  764. };
  765. Application.Run<Toplevel> (null, driver);
  766. #if DEBUG_IDISPOSABLE
  767. Assert.False (Application.Top.WasDisposed);
  768. var exception = Record.Exception (() => Application.Shutdown ());
  769. Assert.NotNull (exception);
  770. Assert.False (Application.Top.WasDisposed);
  771. // It's up to caller to dispose it
  772. Application.Top.Dispose ();
  773. Assert.True (Application.Top.WasDisposed);
  774. #endif
  775. Assert.NotNull (Application.Top);
  776. Application.Shutdown ();
  777. Assert.Null (Application.Top);
  778. }
  779. [Fact]
  780. public void Run_t_Creates_Top_Without_Init ()
  781. {
  782. var driver = new FakeDriver ();
  783. Assert.Null (Application.Top);
  784. Application.Iteration += (s, e) =>
  785. {
  786. Assert.NotNull (Application.Top);
  787. Application.RequestStop ();
  788. };
  789. Application.Run (new (), null, driver);
  790. #if DEBUG_IDISPOSABLE
  791. Assert.False (Application.Top.WasDisposed);
  792. var exception = Record.Exception (() => Application.Shutdown ());
  793. Assert.NotNull (exception);
  794. Assert.False (Application.Top.WasDisposed);
  795. // It's up to caller to dispose it
  796. Application.Top.Dispose ();
  797. Assert.True (Application.Top.WasDisposed);
  798. #endif
  799. Assert.NotNull (Application.Top);
  800. Application.Shutdown ();
  801. Assert.Null (Application.Top);
  802. }
  803. // TODO: Add tests for Run that test errorHandler
  804. #endregion
  805. #region ShutdownTests
  806. [Fact]
  807. public async void Shutdown_Allows_Async ()
  808. {
  809. var isCompletedSuccessfully = false;
  810. async Task TaskWithAsyncContinuation ()
  811. {
  812. await Task.Yield ();
  813. await Task.Yield ();
  814. isCompletedSuccessfully = true;
  815. }
  816. Init ();
  817. Application.Shutdown ();
  818. Assert.False (isCompletedSuccessfully);
  819. await TaskWithAsyncContinuation ();
  820. Thread.Sleep (100);
  821. Assert.True (isCompletedSuccessfully);
  822. }
  823. [Fact]
  824. public void Shutdown_Resets_SyncContext ()
  825. {
  826. Init ();
  827. Application.Shutdown ();
  828. Assert.Null (SynchronizationContext.Current);
  829. }
  830. #endregion
  831. }