Files
Terminal.Gui/UnitTests/ApplicationTests.cs
Thomas Nind ea7981dc59 Adds Key Binding support. Also refactors Autocomplete and Undo/Redo. (#1556)
* Refactored ProcessKey to use public methods for case logic

* Added KeyBinding class

* Refactored key binding to split key->command from command->implementation

This reduces duplication and simplifies the API

* Finishing key bindings implementation in ListView.

* Adding more unit tests to the ListView.

* Added key bindings to the Button and more features.

* Replaces Action for Func<KeyEvent, bool> on CommandImplementations.

* Allowing commands to have any number of arguments.

* Implementing key bindings on Checkbox view.

* Added test for changing HotKey in Button and made ReplaceKeyBinding protected

* Changed `CommandImplementations` to `Func<KeyEvent, bool>` to better understand current command implementations

* Implementing key bindings in ComboBox.

* Renamed Command keys and fixed ComboBox issues:

- Fixed pressing Esc in ListAndCombos scenario without selecting cause an array out of bounds error
- Changed the Esc key in ComboBox to also collapse the list selection
- Added bool return to public virtual method Expand and Collapse (this is a breaking change)

* Implementing key bindings in DateField.

* Organizing some things.

* Implementing key bindings on TimeField.

* No key bindings on FrameView.

* Added keybinding support to TreeView

* Added mouse support and more features.

* Updating NuGet packages.

* Putting text on the same line.

* Changing function command to Func<bool>.

* Added a read only Position, CursorPosition properties and events.

* Keybindings for GraphView

* Added a stream argument to ApplyEdits to only save the edits.

* Implementing key bindings on the HexView.

* Added MenuOpened event and others bug fixes.

* Fixing typo.

* Unifying constructors initializations.

* Implementing keybindings in the Menu.

* Removing unnecessary variable.

* Implementing keybindings in RadioGroup view.

* Changing Home to TopHome and End to BottomEnd.

* Implementing keybindings in the ScrollView.

* Changing the PageLeft and PageRight keybindings.

* Fixing PageLeft and RightPage.

* Removing CleanUp command.

* Key bindings for TabView

* Keybindings for TableView

* Fixed unit tests for PageDown to correctly assign input focus to the TableView

* Fixes the CalculateLeftColumn method avoiding jump two columns on forward moving.

* Fixes #1525. Gives the same backspace behavior as TextView.

* Changes kill-to-start key to work on Linux too.

* Fixes SelectedStart, SelectedText and some cleaning.

* Implementing keybindings in TextField.

* Updated command names and merged as discussed with @BDisp

- Merged LeftItem and LeftChar to Left (same for Right).
- Also renamed Kill to Cut
- Added ScrollLeft / ScrollRight (and renamed ScrollLineUp to just ScrollUp

* Renamed Command.InsertChar to ToggleOverwriteMode and added Enable/Disable

* Removed 'Mode' suffix from toggle overwrite

* Allows navigation to outside a TextView if IsMdiContainer is true.

* Implementing keybindings in Toplevel.

* Fixing null reference exception.

* Changing to keys instances events instead static.

* Transferring the events to the Toplevel.

* Implementing keybindings in TextView.

* Removing static from the QuitKeyChanged and adding unit test.

* Replacing Added with the Initialized event.

* Ignore control characters and other special keys.

* Changing InvokeKeybindings to return Nullable bool and added two more keys to the Toplevel.

* Implementing keybindings in Autocomplete. I had to derive from View.

* Added keybindings menu item to UICatalog

* Added ClearBinding

* Implementing IAutocomplete, abstract Autocomplete and derived TextViewAutocomplete.

* Implementing keybindings in the TextValidateProvider

* Add keybinding to CellActivationKey.

* Fixing some formats.

* Add ObjectActivationKey to the keybindings.

* Made it much easier to implement abstract base `Autocomplete` in other views by moving methods up out of `TextViewAutocomplete` implementation

* Allowing Autocomplete to popup inside or outside the container.

* Fixes the cursor not being showing if the text length is equal to the view width.

* A unit test to prove the 4df5897.

* Removed unused method `GetCursorPosition` from Autocomplete

* Trimmed down implementation specific methods from IAutocomplete

* Fixed xmldoc comment tag

* Format Autocomplete on multiline and fixes wrap settings.

* Adding keys from a to z to avoid the Key.Space on ToString.

* Fixes the vertical position outside the container.

* Adding more key unit tests.

* Changing comment to upper case and proving that doesn't will breaking nothing.

* Replaces Pos.Bottom to Pos.AnchorEnd.

* Fixes popup on resizing.

* Should only using the Pos.Bottom to position outside the view.

* Fixes #1584

* Fixes https://github.com/migueldeicaza/gui.cs/issues/1584#issuecomment-1027987475

* Fixes some bugs with SelectedItem.

* Command must also return a nullable bool.

* Ensures updating the ComboBox text on leaving the control.

* Only with the nullable bool was possible to make the MoveUp and the MoveDown working.

* Added logging of which scenario failed in test

Co-authored-by: BDisp <bd.bdisp@gmail.com>
2022-02-08 10:40:40 -08:00

1284 lines
35 KiB
C#

using System;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Xunit;
// Alias Console to MockConsole so we don't accidentally use Console
using Console = Terminal.Gui.FakeConsole;
namespace Terminal.Gui.Core {
public class ApplicationTests {
public ApplicationTests ()
{
#if DEBUG_IDISPOSABLE
Responder.Instances.Clear ();
#endif
}
[Fact]
public void Init_Shutdown_Cleans_Up ()
{
// Verify initial state is per spec
Pre_Init_State ();
Application.Init (new FakeDriver (), new FakeMainLoop (() => FakeConsole.ReadKey (true)));
// Verify post-Init state is correct
Post_Init_State ();
// MockDriver is always 80x25
Assert.Equal (80, Application.Driver.Cols);
Assert.Equal (25, Application.Driver.Rows);
Application.Shutdown ();
// Verify state is back to initial
Pre_Init_State ();
}
void Pre_Init_State ()
{
Assert.Null (Application.Driver);
Assert.Null (Application.Top);
Assert.Null (Application.Current);
Assert.Throws<ArgumentNullException> (() => Application.HeightAsBuffer == true);
Assert.False (Application.AlwaysSetPosition);
Assert.Null (Application.MainLoop);
Assert.Null (Application.Iteration);
Assert.Null (Application.RootMouseEvent);
Assert.Null (Application.Resized);
}
void Post_Init_State ()
{
Assert.NotNull (Application.Driver);
Assert.NotNull (Application.Top);
Assert.NotNull (Application.Current);
Assert.False (Application.HeightAsBuffer);
Assert.False (Application.AlwaysSetPosition);
Assert.NotNull (Application.MainLoop);
Assert.Null (Application.Iteration);
Assert.Null (Application.RootMouseEvent);
Assert.Null (Application.Resized);
}
[Fact]
public void RunState_Dispose_Cleans_Up ()
{
var rs = new Application.RunState (null);
Assert.NotNull (rs);
// Should not throw because Toplevel was null
rs.Dispose ();
var top = new Toplevel ();
rs = new Application.RunState (top);
Assert.NotNull (rs);
// Should throw because there's no stack
Assert.Throws<InvalidOperationException> (() => rs.Dispose ());
}
void Init ()
{
Application.Init (new FakeDriver (), new FakeMainLoop (() => FakeConsole.ReadKey (true)));
Assert.NotNull (Application.Driver);
Assert.NotNull (Application.MainLoop);
}
void Shutdown ()
{
Application.Shutdown ();
}
[Fact]
public void Begin_End_Cleana_Up ()
{
// Setup Mock driver
Init ();
// Test null Toplevel
Assert.Throws<ArgumentNullException> (() => Application.Begin (null));
var top = new Toplevel ();
var rs = Application.Begin (top);
Assert.NotNull (rs);
Assert.Equal (top, Application.Current);
Application.End (rs);
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 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 = () => {
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 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 = () => {
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 KeyUp_Event ()
{
// Setup Mock driver
Init ();
// Setup some fake keypresses (This)
var input = "Tests";
// Put a control-q in at the end
Console.MockKeyPresses.Push (new ConsoleKeyInfo ('q', ConsoleKey.Q, shift: false, alt: false, control: true));
foreach (var c in input.Reverse ()) {
if (char.IsLetter (c)) {
Console.MockKeyPresses.Push (new ConsoleKeyInfo (c, (ConsoleKey)char.ToUpper (c), shift: char.IsUpper (c), alt: false, control: false));
} else {
Console.MockKeyPresses.Push (new ConsoleKeyInfo (c, (ConsoleKey)c, shift: false, alt: false, control: false));
}
}
int stackSize = Console.MockKeyPresses.Count;
int iterations = 0;
Application.Iteration = () => {
iterations++;
// Stop if we run out of control...
if (iterations > 10) {
Application.RequestStop ();
}
};
int keyUps = 0;
var output = string.Empty;
Application.Top.KeyUp += (View.KeyEventEventArgs args) => {
if (args.KeyEvent.Key != (Key.CtrlMask | Key.Q)) {
output += (char)args.KeyEvent.KeyValue;
}
keyUps++;
};
Application.Run (Application.Top);
// Input string should match output
Assert.Equal (input, output);
// # of key up events should match stack size
//Assert.Equal (stackSize, keyUps);
// We can't use numbers variables on the left side of an Assert.Equal/NotEqual,
// it must be literal (Linux only).
Assert.Equal (6, keyUps);
// # of key up events should match # of iterations
Assert.Equal (stackSize, iterations);
Application.Shutdown ();
Assert.Null (Application.Current);
Assert.Null (Application.Top);
Assert.Null (Application.MainLoop);
Assert.Null (Application.Driver);
}
[Fact]
public void Loaded_Ready_Unlodaded_Events ()
{
Init ();
var top = Application.Top;
var count = 0;
top.Loaded += () => count++;
top.Ready += () => count++;
top.Unloaded += () => count++;
Application.Iteration = () => Application.RequestStop ();
Application.Run ();
Application.Shutdown ();
Assert.Equal (3, count);
}
[Fact]
public void Shutdown_Allows_Async ()
{
static async Task TaskWithAsyncContinuation ()
{
await Task.Yield ();
await Task.Yield ();
}
Init ();
Application.Shutdown ();
var task = TaskWithAsyncContinuation ();
Thread.Sleep (20);
Assert.True (task.IsCompletedSuccessfully);
}
[Fact]
public void Shutdown_Resets_SyncContext ()
{
Init ();
Application.Shutdown ();
Assert.Null (SynchronizationContext.Current);
}
[Fact]
public void AlternateForwardKey_AlternateBackwardKey_Tests ()
{
Init ();
var top = Application.Top;
var w1 = new Window ();
var v1 = new TextField ();
var v2 = new TextView ();
w1.Add (v1, v2);
var w2 = new Window ();
var v3 = new CheckBox ();
var v4 = new Button ();
w2.Add (v3, v4);
top.Add (w1, w2);
Application.Iteration += () => {
Assert.True (v1.HasFocus);
// Using default keys.
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab,
new KeyModifiers () { Ctrl = true }));
Assert.True (v2.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab,
new KeyModifiers () { Ctrl = true }));
Assert.True (v3.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab,
new KeyModifiers () { Ctrl = true }));
Assert.True (v4.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab,
new KeyModifiers () { Ctrl = true }));
Assert.True (v1.HasFocus);
top.ProcessKey (new KeyEvent (Key.ShiftMask | Key.CtrlMask | Key.Tab,
new KeyModifiers () { Shift = true, Ctrl = true }));
Assert.True (v4.HasFocus);
top.ProcessKey (new KeyEvent (Key.ShiftMask | Key.CtrlMask | Key.Tab,
new KeyModifiers () { Shift = true, Ctrl = true }));
Assert.True (v3.HasFocus);
top.ProcessKey (new KeyEvent (Key.ShiftMask | Key.CtrlMask | Key.Tab,
new KeyModifiers () { Shift = true, Ctrl = true }));
Assert.True (v2.HasFocus);
top.ProcessKey (new KeyEvent (Key.ShiftMask | Key.CtrlMask | Key.Tab,
new KeyModifiers () { Shift = true, Ctrl = true }));
Assert.True (v1.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageDown,
new KeyModifiers () { Ctrl = true }));
Assert.True (v2.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageDown,
new KeyModifiers () { Ctrl = true }));
Assert.True (v3.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageDown,
new KeyModifiers () { Ctrl = true }));
Assert.True (v4.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageDown,
new KeyModifiers () { Ctrl = true }));
Assert.True (v1.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageUp,
new KeyModifiers () { Ctrl = true }));
Assert.True (v4.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageUp,
new KeyModifiers () { Ctrl = true }));
Assert.True (v3.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageUp,
new KeyModifiers () { Ctrl = true }));
Assert.True (v2.HasFocus);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.PageUp,
new KeyModifiers () { Ctrl = true }));
Assert.True (v1.HasFocus);
// Using another's alternate keys.
Application.AlternateForwardKey = Key.F7;
Application.AlternateBackwardKey = Key.F6;
top.ProcessKey (new KeyEvent (Key.F7, new KeyModifiers ()));
Assert.True (v2.HasFocus);
top.ProcessKey (new KeyEvent (Key.F7, new KeyModifiers ()));
Assert.True (v3.HasFocus);
top.ProcessKey (new KeyEvent (Key.F7, new KeyModifiers ()));
Assert.True (v4.HasFocus);
top.ProcessKey (new KeyEvent (Key.F7, new KeyModifiers ()));
Assert.True (v1.HasFocus);
top.ProcessKey (new KeyEvent (Key.F6, new KeyModifiers ()));
Assert.True (v4.HasFocus);
top.ProcessKey (new KeyEvent (Key.F6, new KeyModifiers ()));
Assert.True (v3.HasFocus);
top.ProcessKey (new KeyEvent (Key.F6, new KeyModifiers ()));
Assert.True (v2.HasFocus);
top.ProcessKey (new KeyEvent (Key.F6, new KeyModifiers ()));
Assert.True (v1.HasFocus);
Application.RequestStop ();
};
Application.Run (top);
// Replacing the defaults keys to avoid errors on others unit tests that are using it.
Application.AlternateForwardKey = Key.PageDown | Key.CtrlMask;
Application.AlternateBackwardKey = Key.PageUp | Key.CtrlMask;
Application.QuitKey = Key.Q | Key.CtrlMask;
Assert.Equal (Key.PageDown | Key.CtrlMask, Application.AlternateForwardKey);
Assert.Equal (Key.PageUp | Key.CtrlMask, Application.AlternateBackwardKey);
Assert.Equal (Key.Q | Key.CtrlMask, Application.QuitKey);
// Shutdown must be called to safely clean up Application if Init has been called
Application.Shutdown ();
}
[Fact]
public void Application_RequestStop_With_Params_On_A_Not_MdiContainer_Always_Use_The_Application_Current ()
{
Init ();
var top1 = new Toplevel ();
var top2 = new Toplevel ();
var top3 = new Window ();
var top4 = new Window ();
var d = new Dialog ();
// top1, top2, top3, d1 = 4
var iterations = 4;
top1.Ready += () => {
Assert.Null (Application.MdiChildes);
Application.Run (top2);
};
top2.Ready += () => {
Assert.Null (Application.MdiChildes);
Application.Run (top3);
};
top3.Ready += () => {
Assert.Null (Application.MdiChildes);
Application.Run (top4);
};
top4.Ready += () => {
Assert.Null (Application.MdiChildes);
Application.Run (d);
};
d.Ready += () => {
Assert.Null (Application.MdiChildes);
// This will close the d because on a not MdiContainer the Application.Current it always used.
Application.RequestStop (top1);
Assert.True (Application.Current == d);
};
d.Closed += (e) => Application.RequestStop (top1);
Application.Iteration += () => {
Assert.Null (Application.MdiChildes);
if (iterations == 4) {
Assert.True (Application.Current == d);
} else if (iterations == 3) {
Assert.True (Application.Current == top4);
} else if (iterations == 2) {
Assert.True (Application.Current == top3);
} else if (iterations == 1) {
Assert.True (Application.Current == top2);
} else {
Assert.True (Application.Current == top1);
}
Application.RequestStop (top1);
iterations--;
};
Application.Run (top1);
Assert.Null (Application.MdiChildes);
Application.Shutdown ();
}
class Mdi : Toplevel {
public Mdi ()
{
IsMdiContainer = true;
}
}
[Fact]
public void MdiContainer_With_Toplevel_RequestStop_Balanced ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
var d = new Dialog ();
// MdiChild = c1, c2, c3
// d1 = 1
var iterations = 4;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (d);
};
// More easy because the Mdi Container handles all at once
d.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
// This will not close the MdiContainer because d is a modal toplevel and will be closed.
mdi.RequestStop ();
};
// Now this will close the MdiContainer propagating through the MdiChildes.
d.Closed += (e) => {
mdi.RequestStop ();
};
Application.Iteration += () => {
if (iterations == 4) {
// The Dialog was not closed before and will be closed now.
Assert.True (Application.Current == d);
Assert.False (d.Running);
} else {
Assert.Equal (iterations, Application.MdiChildes.Count);
for (int i = 0; i < iterations; i++) {
Assert.Equal ((iterations - i + 1).ToString (), Application.MdiChildes [i].Id);
}
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void MdiContainer_With_Application_RequestStop_MdiTop_With_Params ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
var d = new Dialog ();
// MdiChild = c1, c2, c3
// d1 = 1
var iterations = 4;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (d);
};
// Also easy because the Mdi Container handles all at once
d.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
// This will not close the MdiContainer because d is a modal toplevel
Application.RequestStop (mdi);
};
// Now this will close the MdiContainer propagating through the MdiChildes.
d.Closed += (e) => Application.RequestStop (mdi);
Application.Iteration += () => {
if (iterations == 4) {
// The Dialog was not closed before and will be closed now.
Assert.True (Application.Current == d);
Assert.False (d.Running);
} else {
Assert.Equal (iterations, Application.MdiChildes.Count);
for (int i = 0; i < iterations; i++) {
Assert.Equal ((iterations - i + 1).ToString (), Application.MdiChildes [i].Id);
}
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void MdiContainer_With_Application_RequestStop_MdiTop_Without_Params ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
var d = new Dialog ();
// MdiChild = c1, c2, c3 = 3
// d1 = 1
var iterations = 4;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (d);
};
//More harder because it's sequential.
d.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
// Close the Dialog
Application.RequestStop ();
};
// Now this will close the MdiContainer propagating through the MdiChildes.
d.Closed += (e) => Application.RequestStop (mdi);
Application.Iteration += () => {
if (iterations == 4) {
// The Dialog still is the current top and we can't request stop to MdiContainer
// because we are not using parameter calls.
Assert.True (Application.Current == d);
Assert.False (d.Running);
} else {
Assert.Equal (iterations, Application.MdiChildes.Count);
for (int i = 0; i < iterations; i++) {
Assert.Equal ((iterations - i + 1).ToString (), Application.MdiChildes [i].Id);
}
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void IsMdiChild_Testing ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
var d = new Dialog ();
Application.Iteration += () => {
Assert.False (mdi.IsMdiChild);
Assert.True (c1.IsMdiChild);
Assert.True (c2.IsMdiChild);
Assert.True (c3.IsMdiChild);
Assert.False (d.IsMdiChild);
mdi.RequestStop ();
};
Application.Run (mdi);
Application.Shutdown ();
}
[Fact]
public void Modal_Toplevel_Can_Open_Another_Modal_Toplevel_But_RequestStop_To_The_Caller_Also_Sets_Current_Running_To_False_Too ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
var d1 = new Dialog ();
var d2 = new Dialog ();
// MdiChild = c1, c2, c3 = 3
// d1, d2 = 2
var iterations = 5;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (d1);
};
d1.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (d2);
};
d2.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Assert.True (Application.Current == d2);
Assert.True (Application.Current.Running);
// Trying to close the Dialog1
d1.RequestStop ();
};
// Now this will close the MdiContainer propagating through the MdiChildes.
d1.Closed += (e) => {
Assert.True (Application.Current == d1);
Assert.False (Application.Current.Running);
mdi.RequestStop ();
};
Application.Iteration += () => {
if (iterations == 5) {
// The Dialog2 still is the current top and we can't request stop to MdiContainer
// because Dialog2 and Dialog1 must be closed first.
// Dialog2 will be closed in this iteration.
Assert.True (Application.Current == d2);
Assert.False (Application.Current.Running);
Assert.False (d1.Running);
} else if (iterations == 4) {
// Dialog1 will be closed in this iteration.
Assert.True (Application.Current == d1);
Assert.False (Application.Current.Running);
} else {
Assert.Equal (iterations, Application.MdiChildes.Count);
for (int i = 0; i < iterations; i++) {
Assert.Equal ((iterations - i + 1).ToString (), Application.MdiChildes [i].Id);
}
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void Modal_Toplevel_Can_Open_Another_Not_Modal_Toplevel_But_RequestStop_To_The_Caller_Also_Sets_Current_Running_To_False_Too ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
var d1 = new Dialog ();
var c4 = new Toplevel ();
// MdiChild = c1, c2, c3, c4 = 4
// d1 = 1
var iterations = 5;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (d1);
};
d1.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
Application.Run (c4);
};
c4.Ready += () => {
Assert.Equal (4, Application.MdiChildes.Count);
// Trying to close the Dialog1
d1.RequestStop ();
};
// Now this will close the MdiContainer propagating through the MdiChildes.
d1.Closed += (e) => {
mdi.RequestStop ();
};
Application.Iteration += () => {
if (iterations == 5) {
// The Dialog2 still is the current top and we can't request stop to MdiContainer
// because Dialog2 and Dialog1 must be closed first.
// Using request stop here will call the Dialog again without need
Assert.True (Application.Current == d1);
Assert.False (Application.Current.Running);
Assert.True (c4.Running);
} else {
Assert.Equal (iterations, Application.MdiChildes.Count);
for (int i = 0; i < iterations; i++) {
Assert.Equal ((iterations - i + (iterations == 4 && i == 0 ? 2 : 1)).ToString (),
Application.MdiChildes [i].Id);
}
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void MoveCurrent_Returns_False_If_The_Current_And_Top_Parameter_Are_Both_With_Running_Set_To_False ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
// MdiChild = c1, c2, c3
var iterations = 3;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
c3.RequestStop ();
c1.RequestStop ();
};
// Now this will close the MdiContainer propagating through the MdiChildes.
c1.Closed += (e) => {
mdi.RequestStop ();
};
Application.Iteration += () => {
if (iterations == 3) {
// The Current still is c3 because Current.Running is false.
Assert.True (Application.Current == c3);
Assert.False (Application.Current.Running);
// But the childes order were reorder by Running = false
Assert.True (Application.MdiChildes [0] == c3);
Assert.True (Application.MdiChildes [1] == c1);
Assert.True (Application.MdiChildes [^1] == c2);
} else if (iterations == 2) {
// The Current is c1 and Current.Running is false.
Assert.True (Application.Current == c1);
Assert.False (Application.Current.Running);
Assert.True (Application.MdiChildes [0] == c1);
Assert.True (Application.MdiChildes [^1] == c2);
} else if (iterations == 1) {
// The Current is c2 and Current.Running is false.
Assert.True (Application.Current == c2);
Assert.False (Application.Current.Running);
Assert.True (Application.MdiChildes [^1] == c2);
} else {
// The Current is mdi.
Assert.True (Application.Current == mdi);
Assert.Empty (Application.MdiChildes);
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void MdiContainer_Throws_If_More_Than_One ()
{
Init ();
var mdi = new Mdi ();
var mdi2 = new Mdi ();
mdi.Ready += () => {
Assert.Throws<InvalidOperationException> (() => Application.Run (mdi2));
mdi.RequestStop ();
};
Application.Run (mdi);
Application.Shutdown ();
}
[Fact]
public void MdiContainer_Open_And_Close_Modal_And_Open_Not_Modal_Toplevels_Randomly ()
{
Init ();
var mdi = new Mdi ();
var logger = new Toplevel ();
var iterations = 1; // The logger
var running = true;
var stageCompleted = true;
var allStageClosed = false;
var mdiRequestStop = false;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (logger);
};
logger.Ready += () => Assert.Single (Application.MdiChildes);
Application.Iteration += () => {
if (stageCompleted && running) {
stageCompleted = false;
var stage = new Window () { Modal = true };
stage.Ready += () => {
Assert.Equal (iterations, Application.MdiChildes.Count);
stage.RequestStop ();
};
stage.Closed += (_) => {
if (iterations == 11) {
allStageClosed = true;
}
Assert.Equal (iterations, Application.MdiChildes.Count);
if (running) {
stageCompleted = true;
var rpt = new Window ();
rpt.Ready += () => {
iterations++;
Assert.Equal (iterations, Application.MdiChildes.Count);
};
Application.Run (rpt);
}
};
Application.Run (stage);
} else if (iterations == 11 && running) {
running = false;
Assert.Equal (iterations, Application.MdiChildes.Count);
} else if (!mdiRequestStop && running && !allStageClosed) {
Assert.Equal (iterations, Application.MdiChildes.Count);
} else if (!mdiRequestStop && !running && allStageClosed) {
Assert.Equal (iterations, Application.MdiChildes.Count);
mdiRequestStop = true;
mdi.RequestStop ();
} else {
Assert.Empty (Application.MdiChildes);
}
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void AllChildClosed_Event_Test ()
{
Init ();
var mdi = new Mdi ();
var c1 = new Toplevel ();
var c2 = new Window ();
var c3 = new Window ();
// MdiChild = c1, c2, c3
var iterations = 3;
mdi.Ready += () => {
Assert.Empty (Application.MdiChildes);
Application.Run (c1);
};
c1.Ready += () => {
Assert.Single (Application.MdiChildes);
Application.Run (c2);
};
c2.Ready += () => {
Assert.Equal (2, Application.MdiChildes.Count);
Application.Run (c3);
};
c3.Ready += () => {
Assert.Equal (3, Application.MdiChildes.Count);
c3.RequestStop ();
c2.RequestStop ();
c1.RequestStop ();
};
// Now this will close the MdiContainer when all MdiChildes was closed
mdi.AllChildClosed += () => {
mdi.RequestStop ();
};
Application.Iteration += () => {
if (iterations == 3) {
// The Current still is c3 because Current.Running is false.
Assert.True (Application.Current == c3);
Assert.False (Application.Current.Running);
// But the childes order were reorder by Running = false
Assert.True (Application.MdiChildes [0] == c3);
Assert.True (Application.MdiChildes [1] == c2);
Assert.True (Application.MdiChildes [^1] == c1);
} else if (iterations == 2) {
// The Current is c2 and Current.Running is false.
Assert.True (Application.Current == c2);
Assert.False (Application.Current.Running);
Assert.True (Application.MdiChildes [0] == c2);
Assert.True (Application.MdiChildes [^1] == c1);
} else if (iterations == 1) {
// The Current is c1 and Current.Running is false.
Assert.True (Application.Current == c1);
Assert.False (Application.Current.Running);
Assert.True (Application.MdiChildes [^1] == c1);
} else {
// The Current is mdi.
Assert.True (Application.Current == mdi);
Assert.False (Application.Current.Running);
Assert.Empty (Application.MdiChildes);
}
iterations--;
};
Application.Run (mdi);
Assert.Empty (Application.MdiChildes);
Application.Shutdown ();
}
[Fact]
public void SetCurrentAsTop_Run_A_Not_Modal_Toplevel_Make_It_The_Current_Application_Top ()
{
Init ();
var t1 = new Toplevel ();
var t2 = new Toplevel ();
var t3 = new Toplevel ();
var d = new Dialog ();
var t4 = new Toplevel ();
// t1, t2, t3, d, t4
var iterations = 5;
t1.Ready += () => {
Assert.Equal (t1, Application.Top);
Application.Run (t2);
};
t2.Ready += () => {
Assert.Equal (t2, Application.Top);
Application.Run (t3);
};
t3.Ready += () => {
Assert.Equal (t3, Application.Top);
Application.Run (d);
};
d.Ready += () => {
Assert.Equal (t3, Application.Top);
Application.Run (t4);
};
t4.Ready += () => {
Assert.Equal (t4, Application.Top);
t4.RequestStop ();
d.RequestStop ();
t3.RequestStop ();
t2.RequestStop ();
};
// Now this will close the MdiContainer when all MdiChildes was closed
t2.Closed += (_) => {
t1.RequestStop ();
};
Application.Iteration += () => {
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);
Application.Shutdown ();
Assert.Null (Application.Top);
}
[Fact]
[AutoInitShutdown]
public void Internal_Tests ()
{
Assert.True (Application._initialized);
Assert.NotNull (Application.Top);
var rs = Application.Begin (Application.Top);
Assert.Equal (Application.Top, rs.Toplevel);
Assert.Null (Application.mouseGrabView);
Assert.Null (Application.wantContinuousButtonPressedView);
Assert.False (Application.DebugDrawBounds);
Assert.False (Application.ShowChild (Application.Top));
Application.End (Application.Top);
}
[Fact]
[AutoInitShutdown]
public void QuitKey_Getter_Setter ()
{
var top = Application.Top;
var isQuiting = false;
top.Closing += (e) => {
isQuiting = true;
e.Cancel = true;
};
Application.Begin (top);
top.Running = true;
Assert.Equal (Key.Q | Key.CtrlMask, Application.QuitKey);
Application.Driver.SendKeys ('q', ConsoleKey.Q, false, false, true);
Assert.True (isQuiting);
isQuiting = false;
Application.QuitKey = Key.C | Key.CtrlMask;
Application.Driver.SendKeys ('q', ConsoleKey.Q, false, false, true);
Assert.False (isQuiting);
Application.Driver.SendKeys ('c', ConsoleKey.C, false, false, true);
Assert.True (isQuiting);
// Reset the QuitKey to avoid throws errors on another tests
Application.QuitKey = Key.Q | Key.CtrlMask;
}
[Fact]
[AutoInitShutdown]
public void EnsuresTopOnFront_CanFocus_True_By_Keyboard_And_Mouse ()
{
var top = Application.Top;
var win = new Window ("win") { X = 0, Y = 0, Width = 20, Height = 10 };
var tf = new TextField () { Width = 10 };
win.Add (tf);
var win2 = new Window ("win2") { X = 22, Y = 0, Width = 20, Height = 10 };
var tf2 = new TextField () { Width = 10 };
win2.Add (tf2);
top.Add (win, win2);
Application.Begin (top);
Assert.True (win.CanFocus);
Assert.True (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.False (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab, new KeyModifiers ()));
Assert.True (win.CanFocus);
Assert.False (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.True (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab, new KeyModifiers ()));
Assert.True (win.CanFocus);
Assert.True (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.False (win2.HasFocus);
Assert.Equal ("win", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
win2.MouseEvent (new MouseEvent () { Flags = MouseFlags.Button1Pressed });
Assert.True (win.CanFocus);
Assert.False (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.True (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
win2.MouseEvent (new MouseEvent () { Flags = MouseFlags.Button1Released });
Assert.Null (Toplevel.dragPosition);
}
[Fact]
[AutoInitShutdown]
public void EnsuresTopOnFront_CanFocus_False_By_Keyboard_And_Mouse ()
{
var top = Application.Top;
var win = new Window ("win") { X = 0, Y = 0, Width = 20, Height = 10 };
var tf = new TextField () { Width = 10 };
win.Add (tf);
var win2 = new Window ("win2") { X = 22, Y = 0, Width = 20, Height = 10 };
var tf2 = new TextField () { Width = 10 };
win2.Add (tf2);
top.Add (win, win2);
Application.Begin (top);
Assert.True (win.CanFocus);
Assert.True (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.False (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
win.CanFocus = false;
Assert.False (win.CanFocus);
Assert.False (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.True (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab, new KeyModifiers ()));
Assert.True (win2.CanFocus);
Assert.False (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.True (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
top.ProcessKey (new KeyEvent (Key.CtrlMask | Key.Tab, new KeyModifiers ()));
Assert.False (win.CanFocus);
Assert.False (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.True (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
win.MouseEvent (new MouseEvent () { Flags = MouseFlags.Button1Pressed });
Assert.False (win.CanFocus);
Assert.False (win.HasFocus);
Assert.True (win2.CanFocus);
Assert.True (win2.HasFocus);
Assert.Equal ("win2", ((Window)top.Subviews [top.Subviews.Count - 1]).Title);
win2.MouseEvent (new MouseEvent () { Flags = MouseFlags.Button1Released });
Assert.Null (Toplevel.dragPosition);
}
}
}