123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685 |
- using System;
- using System.Diagnostics;
- using System.IO;
- using System.Linq;
- using System.Runtime.InteropServices;
- using System.Threading;
- using System.Threading.Tasks;
- using Xunit;
- using Xunit.Abstractions;
- // Alias Console to MockConsole so we don't accidentally use Console
- using Console = Terminal.Gui.FakeConsole;
- namespace Terminal.Gui.ApplicationTests;
- public class ApplicationTests {
- readonly ITestOutputHelper _output;
- public ApplicationTests (ITestOutputHelper output)
- {
- _output = output;
- ConsoleDriver.RunningUnitTests = true;
- #if DEBUG_IDISPOSABLE
- Responder.Instances.Clear ();
- RunState.Instances.Clear ();
- #endif
- }
- void Pre_Init_State ()
- {
- Assert.Null (Application.Driver);
- Assert.Null (Application.Top);
- Assert.Null (Application.Current);
- Assert.Null (Application.MainLoop);
- }
- void Post_Init_State ()
- {
- Assert.NotNull (Application.Driver);
- Assert.NotNull (Application.Top);
- Assert.NotNull (Application.Current);
- Assert.NotNull (Application.MainLoop);
- // FakeDriver is always 80x25
- Assert.Equal (80, Application.Driver.Cols);
- Assert.Equal (25, Application.Driver.Rows);
- }
- void Init ()
- {
- Application.Init (new FakeDriver ());
- Assert.NotNull (Application.Driver);
- Assert.NotNull (Application.MainLoop);
- Assert.NotNull (SynchronizationContext.Current);
- }
- void Shutdown () => Application.Shutdown ();
- [Fact]
- public void Init_Shutdown_Cleans_Up ()
- {
- // Verify initial state is per spec
- //Pre_Init_State ();
- Application.Init (new FakeDriver ());
- // Verify post-Init state is correct
- //Post_Init_State ();
- Application.Shutdown ();
- // Verify state is back to initial
- //Pre_Init_State ();
- #if DEBUG_IDISPOSABLE
- // Validate there are no outstanding Responder-based instances
- // after a scenario was selected to run. This proves the main UI Catalog
- // 'app' closed cleanly.
- Assert.Empty (Responder.Instances);
- #endif
- }
- [Fact]
- public void Init_Unbalanced_Throws ()
- {
- Application.Init (new FakeDriver ());
- Toplevel topLevel = null;
- Assert.Throws<InvalidOperationException> (() => Application.InternalInit (() => topLevel = new TestToplevel (), new FakeDriver ()));
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- // Now try the other way
- topLevel = null;
- Application.InternalInit (() => topLevel = new TestToplevel (), new FakeDriver ());
- Assert.Throws<InvalidOperationException> (() => Application.Init (new FakeDriver ()));
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- class TestToplevel : Toplevel {
- public TestToplevel () => IsOverlappedContainer = false;
- }
- [Fact]
- public void Init_Null_Driver_Should_Pick_A_Driver ()
- {
- Application.Init (null);
- Assert.NotNull (Application.Driver);
- Shutdown ();
- }
- [Theory]
- [InlineData (typeof (FakeDriver))]
- [InlineData (typeof (NetDriver))]
- //[InlineData (typeof (ANSIDriver))]
- [InlineData (typeof (WindowsDriver))]
- [InlineData (typeof (CursesDriver))]
- public void Init_DriverName_Should_Pick_Correct_Driver (Type driverType)
- {
- var driver = (ConsoleDriver)Activator.CreateInstance (driverType);
- Application.Init (driverName: driverType.Name);
- Assert.NotNull (Application.Driver);
- Assert.Equal (driverType, Application.Driver.GetType ());
- Shutdown ();
- }
- [Fact]
- public void Init_Begin_End_Cleans_Up ()
- {
- Init ();
- // Begin will cause Run() to be called, which will call Begin(). Thus will block the tests
- // if we don't stop
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- RunState runstate = null;
- EventHandler<RunStateEventArgs> NewRunStateFn = (s, e) => {
- Assert.NotNull (e.State);
- runstate = e.State;
- };
- Application.NotifyNewRunState += NewRunStateFn;
- var topLevel = new Toplevel ();
- var rs = Application.Begin (topLevel);
- Assert.NotNull (rs);
- Assert.NotNull (runstate);
- Assert.Equal (rs, runstate);
- Assert.Equal (topLevel, Application.Top);
- Assert.Equal (topLevel, Application.Current);
- Application.NotifyNewRunState -= NewRunStateFn;
- Application.End (runstate);
- Assert.Null (Application.Current);
- Assert.NotNull (Application.Top);
- Assert.NotNull (Application.MainLoop);
- Assert.NotNull (Application.Driver);
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void InitWithTopLevelFactory_Begin_End_Cleans_Up ()
- {
- // Begin will cause Run() to be called, which will call Begin(). Thus will block the tests
- // if we don't stop
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- // NOTE: Run<T>, when called after Init has been called behaves differently than
- // when called if Init has not been called.
- Toplevel topLevel = null;
- Application.InternalInit (() => topLevel = new TestToplevel (), new FakeDriver ());
- RunState runstate = null;
- EventHandler<RunStateEventArgs> NewRunStateFn = (s, e) => {
- Assert.NotNull (e.State);
- runstate = e.State;
- };
- Application.NotifyNewRunState += NewRunStateFn;
- var rs = Application.Begin (topLevel);
- Assert.NotNull (rs);
- Assert.NotNull (runstate);
- Assert.Equal (rs, runstate);
- Assert.Equal (topLevel, Application.Top);
- Assert.Equal (topLevel, Application.Current);
- Application.NotifyNewRunState -= NewRunStateFn;
- Application.End (runstate);
- Assert.Null (Application.Current);
- Assert.NotNull (Application.Top);
- Assert.NotNull (Application.MainLoop);
- Assert.NotNull (Application.Driver);
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Begin_Null_Toplevel_Throws ()
- {
- // Setup Mock driver
- Init ();
- // Test null Toplevel
- Assert.Throws<ArgumentNullException> (() => Application.Begin (null));
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- #region RunTests
- [Fact]
- public void Run_T_After_InitWithDriver_with_TopLevel_Throws ()
- {
- // Setup Mock driver
- Init ();
- // Run<Toplevel> when already initialized with a Driver will throw (because Toplevel is not dervied from TopLevel)
- Assert.Throws<ArgumentException> (() => Application.Run<Toplevel> (errorHandler: null));
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_T_After_InitWithDriver_with_TopLevel_and_Driver_Throws ()
- {
- // Setup Mock driver
- Init ();
- // Run<Toplevel> when already initialized with a Driver will throw (because Toplevel is not dervied from TopLevel)
- Assert.Throws<ArgumentException> (() => Application.Run<Toplevel> (null, new FakeDriver ()));
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_T_After_InitWithDriver_with_TestTopLevel_DoesNotThrow ()
- {
- // Setup Mock driver
- Init ();
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- // Init has been called and we're passing no driver to Run<TestTopLevel>. This is ok.
- Application.Run<TestToplevel> ();
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_T_After_InitNullDriver_with_TestTopLevel_Throws ()
- {
- Application.ForceDriver = "FakeDriver";
- Application.Init (null);
- Assert.Equal (typeof (FakeDriver), Application.Driver.GetType ());
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- // Init has been called without selecting a driver and we're passing no driver to Run<TestTopLevel>. Bad
- Application.Run<TestToplevel> ();
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_T_Init_Driver_Cleared_with_TestTopLevel_Throws ()
- {
- Init ();
- Application.Driver = null;
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- // Init has been called, but Driver has been set to null. Bad.
- Assert.Throws<InvalidOperationException> (() => Application.Run<TestToplevel> ());
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_T_NoInit_DoesNotThrow ()
- {
- Application.ForceDriver = "FakeDriver";
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- Application.Run<TestToplevel> ();
- Assert.Equal (typeof (FakeDriver), Application.Driver.GetType ());
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_T_NoInit_WithDriver_DoesNotThrow ()
- {
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- // Init has NOT been called and we're passing a valid driver to Run<TestTopLevel>. This is ok.
- Application.Run<TestToplevel> (null, new FakeDriver ());
- Shutdown ();
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_RequestStop_Stops ()
- {
- // Setup Mock driver
- Init ();
- var top = new Toplevel ();
- var rs = Application.Begin (top);
- Assert.NotNull (rs);
- Assert.Equal (top, Application.Current);
- Application.Iteration += (s, a) => {
- Application.RequestStop ();
- };
- Application.Run (top);
- Application.Shutdown ();
- Assert.Null (Application.Current);
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_RunningFalse_Stops ()
- {
- // Setup Mock driver
- Init ();
- var top = new Toplevel ();
- var rs = Application.Begin (top);
- Assert.NotNull (rs);
- Assert.Equal (top, Application.Current);
- Application.Iteration += (s, a) => {
- top.Running = false;
- };
- Application.Run (top);
- Application.Shutdown ();
- Assert.Null (Application.Current);
- Assert.Null (Application.Top);
- Assert.Null (Application.MainLoop);
- Assert.Null (Application.Driver);
- }
- [Fact]
- public void Run_Loaded_Ready_Unlodaded_Events ()
- {
- Init ();
- var top = Application.Top;
- int count = 0;
- top.Loaded += (s, e) => count++;
- top.Ready += (s, e) => count++;
- top.Unloaded += (s, e) => count++;
- Application.Iteration += (s, a) => Application.RequestStop ();
- Application.Run ();
- Application.Shutdown ();
- Assert.Equal (3, count);
- }
- // TODO: All Toplevel layout tests should be moved to ToplevelTests.cs
- [Fact]
- public void Run_Toplevel_With_Modal_View_Does_Not_Refresh_If_Not_Dirty ()
- {
- Init ();
- int count = 0;
- // Don't use Dialog here as it has more layout logic. Use Window instead.
- Dialog d = null;
- var top = Application.Top;
- top.DrawContent += (s, a) => count++;
- int iteration = -1;
- Application.Iteration += (s, a) => {
- iteration++;
- if (iteration == 0) {
- // TODO: Don't use Dialog here as it has more layout logic. Use Window instead.
- d = new Dialog ();
- d.DrawContent += (s, a) => count++;
- Application.Run (d);
- } else if (iteration < 3) {
- Application.OnMouseEvent (new MouseEventEventArgs (new MouseEvent { X = 0, Y = 0, Flags = MouseFlags.ReportMousePosition }));
- Assert.False (top.NeedsDisplay);
- Assert.False (top.SubViewNeedsDisplay);
- Assert.False (top.LayoutNeeded);
- Assert.False (d.NeedsDisplay);
- Assert.False (d.SubViewNeedsDisplay);
- Assert.False (d.LayoutNeeded);
- } else {
- Application.RequestStop ();
- }
- };
- Application.Run ();
- Application.Shutdown ();
- // 1 - First top load, 1 - Dialog load, 1 - Dialog unload, Total - 3.
- Assert.Equal (3, count);
- }
- // TODO: All Toplevel layout tests should be moved to ToplevelTests.cs
- [Fact]
- public void Run_A_Modal_Toplevel_Refresh_Background_On_Moving ()
- {
- Init ();
- // Don't use Dialog here as it has more layout logic. Use Window instead.
- var w = new Window () { Width = 5, Height = 5 };
- ((FakeDriver)Application.Driver).SetBufferSize (10, 10);
- var rs = Application.Begin (w);
- TestHelpers.AssertDriverContentsWithFrameAre (@"
- ┌───┐
- │ │
- │ │
- │ │
- └───┘", _output);
- var attributes = new Attribute [] {
- // 0
- new Attribute (ColorName.White, ColorName.Black),
- // 1
- Colors.Base.Normal
- };
- TestHelpers.AssertDriverAttributesAre (@"
- 1111100000
- 1111100000
- 1111100000
- 1111100000
- 1111100000
- ", null, attributes);
- // TODO: In PR #2920 this breaks because the mouse is not grabbed anymore.
- // TODO: Move the mouse grap/drag mode from Toplevel to Border.
- Application.OnMouseEvent (new MouseEventEventArgs (new MouseEvent () { X = 0, Y = 0, Flags = MouseFlags.Button1Pressed }));
- Assert.Equal (w, Application.MouseGrabView);
- // Move down and to the right.
- Application.OnMouseEvent (new MouseEventEventArgs (new MouseEvent () { X = 1, Y = 1, Flags = MouseFlags.Button1Pressed | MouseFlags.ReportMousePosition }));
- Application.Refresh ();
- TestHelpers.AssertDriverContentsWithFrameAre (@"
- ┌───┐
- │ │
- │ │
- │ │
- └───┘", _output);
- attributes = new Attribute [] {
- // 0
- new Attribute (ColorName.White, ColorName.Black),
- // 1
- Colors.Base.Normal
- };
- TestHelpers.AssertDriverAttributesAre (@"
- 0000000000
- 0111110000
- 0111110000
- 0111110000
- 0111110000
- 0111110000
- ", null, attributes);
- Application.End (rs);
- Application.Shutdown ();
- }
- // TODO: Add tests for Run that test errorHandler
- #endregion
- #region ShutdownTests
- [Fact]
- public async void Shutdown_Allows_Async ()
- {
- bool isCompletedSuccessfully = false;
- async Task TaskWithAsyncContinuation ()
- {
- await Task.Yield ();
- await Task.Yield ();
- isCompletedSuccessfully = true;
- }
- Init ();
- Application.Shutdown ();
- Assert.False (isCompletedSuccessfully);
- await TaskWithAsyncContinuation ();
- Thread.Sleep (100);
- Assert.True (isCompletedSuccessfully);
- }
- [Fact]
- public void Shutdown_Resets_SyncContext ()
- {
- Init ();
- Application.Shutdown ();
- Assert.Null (SynchronizationContext.Current);
- }
- #endregion
- [Fact, AutoInitShutdown]
- public void Begin_Sets_Application_Top_To_Console_Size ()
- {
- Assert.Equal (new Rect (0, 0, 80, 25), Application.Top.Frame);
- Application.Begin (Application.Top);
- Assert.Equal (new Rect (0, 0, 80, 25), Application.Top.Frame);
- ((FakeDriver)Application.Driver).SetBufferSize (5, 5);
- Assert.Equal (new Rect (0, 0, 5, 5), Application.Top.Frame);
- }
- [Fact]
- [AutoInitShutdown]
- public void SetCurrentAsTop_Run_A_Not_Modal_Toplevel_Make_It_The_Current_Application_Top ()
- {
- var t1 = new Toplevel ();
- var t2 = new Toplevel ();
- var t3 = new Toplevel ();
- // Don't use Dialog here as it has more layout logic. Use Window instead.
- var d = new Dialog ();
- var t4 = new Toplevel ();
- // t1, t2, t3, d, t4
- int iterations = 5;
- t1.Ready += (s, e) => {
- Assert.Equal (t1, Application.Top);
- Application.Run (t2);
- };
- t2.Ready += (s, e) => {
- Assert.Equal (t2, Application.Top);
- Application.Run (t3);
- };
- t3.Ready += (s, e) => {
- Assert.Equal (t3, Application.Top);
- Application.Run (d);
- };
- d.Ready += (s, e) => {
- Assert.Equal (t3, Application.Top);
- Application.Run (t4);
- };
- t4.Ready += (s, e) => {
- Assert.Equal (t4, Application.Top);
- t4.RequestStop ();
- d.RequestStop ();
- t3.RequestStop ();
- t2.RequestStop ();
- };
- // Now this will close the OverlappedContainer when all OverlappedChildren was closed
- t2.Closed += (s, _) => {
- t1.RequestStop ();
- };
- Application.Iteration += (s, a) => {
- if (iterations == 5) {
- // The Current still is t4 because Current.Running is false.
- Assert.Equal (t4, Application.Current);
- Assert.False (Application.Current.Running);
- Assert.Equal (t4, Application.Top);
- } else if (iterations == 4) {
- // The Current is d and Current.Running is false.
- Assert.Equal (d, Application.Current);
- Assert.False (Application.Current.Running);
- Assert.Equal (t4, Application.Top);
- } else if (iterations == 3) {
- // The Current is t3 and Current.Running is false.
- Assert.Equal (t3, Application.Current);
- Assert.False (Application.Current.Running);
- Assert.Equal (t3, Application.Top);
- } else if (iterations == 2) {
- // The Current is t2 and Current.Running is false.
- Assert.Equal (t2, Application.Current);
- Assert.False (Application.Current.Running);
- Assert.Equal (t2, Application.Top);
- } else {
- // The Current is t1.
- Assert.Equal (t1, Application.Current);
- Assert.False (Application.Current.Running);
- Assert.Equal (t1, Application.Top);
- }
- iterations--;
- };
- Application.Run (t1);
- Assert.Equal (t1, Application.Top);
- }
- [Fact]
- [AutoInitShutdown]
- public void Internal_Properties_Correct ()
- {
- Assert.True (Application._initialized);
- Assert.NotNull (Application.Top);
- var rs = Application.Begin (Application.Top);
- Assert.Equal (Application.Top, rs.Toplevel);
- Assert.Null (Application.MouseGrabView); // public
- Assert.Null (Application.WantContinuousButtonPressedView); // public
- Assert.False (Application.MoveToOverlappedChild (Application.Top));
- }
- // Invoke Tests
- // TODO: Test with threading scenarios
- [Fact]
- public void Invoke_Adds_Idle ()
- {
- Application.Init (new FakeDriver ());
- var top = new Toplevel ();
- var rs = Application.Begin (top);
- bool firstIteration = false;
- int actionCalled = 0;
- Application.Invoke (() => { actionCalled++; });
- Application.MainLoop.Running = true;
- Application.RunIteration (ref rs, ref firstIteration);
- Assert.Equal (1, actionCalled);
- Application.Shutdown ();
- }
- }
|