123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458 |
- using System;
- using Xunit;
- using Xunit.Abstractions;
- // Alias Console to MockConsole so we don't accidentally use Console
- using Console = Terminal.Gui.FakeConsole;
- namespace Terminal.Gui.ViewTests;
- public class KeyboardEventTests {
- readonly ITestOutputHelper _output;
- public KeyboardEventTests (ITestOutputHelper output) => _output = output;
- [Fact]
- public void KeyPress_Handled_Cancels ()
- {
- var view = new View ();
- bool invokingKeyBindingsInvoked = false;
- bool processKeyPressInvoked = false;
- bool setHandledTo = false;
- view.KeyDown += (s, e) => {
- e.Handled = setHandledTo;
- Assert.Equal (setHandledTo, e.Handled);
- Assert.Equal (KeyCode.N, e.KeyCode);
- };
- view.InvokingKeyBindings += (s, e) => {
- invokingKeyBindingsInvoked = true;
- Assert.False (e.Handled);
- Assert.Equal (KeyCode.N, e.KeyCode);
- };
- view.ProcessKeyDown += (s, e) => {
- processKeyPressInvoked = true;
- Assert.False (e.Handled);
- Assert.Equal (KeyCode.N, e.KeyCode);
- };
- view.NewKeyDownEvent (new Key (KeyCode.N));
- Assert.True (invokingKeyBindingsInvoked);
- Assert.True (processKeyPressInvoked);
- invokingKeyBindingsInvoked = false;
- processKeyPressInvoked = false;
- setHandledTo = true;
- view.NewKeyDownEvent (new Key (KeyCode.N));
- Assert.False (invokingKeyBindingsInvoked);
- Assert.False (processKeyPressInvoked);
- }
- [Fact]
- public void InvokingKeyBindings_Handled_Cancels ()
- {
- var view = new View ();
- bool keyPressInvoked = false;
- bool invokingKeyBindingsInvoked = false;
- bool processKeyPressInvoked = false;
- bool setHandledTo = false;
- view.KeyDown += (s, e) => {
- keyPressInvoked = true;
- Assert.False (e.Handled);
- Assert.Equal (KeyCode.N, e.KeyCode);
- };
- view.InvokingKeyBindings += (s, e) => {
- invokingKeyBindingsInvoked = true;
- e.Handled = setHandledTo;
- Assert.Equal (setHandledTo, e.Handled);
- Assert.Equal (KeyCode.N, e.KeyCode);
- };
- view.ProcessKeyDown += (s, e) => {
- processKeyPressInvoked = true;
- processKeyPressInvoked = true;
- Assert.False (e.Handled);
- Assert.Equal (KeyCode.N, e.KeyCode);
- };
- view.NewKeyDownEvent (new Key (KeyCode.N));
- Assert.True (keyPressInvoked);
- Assert.True (invokingKeyBindingsInvoked);
- Assert.True (processKeyPressInvoked);
- keyPressInvoked = false;
- invokingKeyBindingsInvoked = false;
- processKeyPressInvoked = false;
- setHandledTo = true;
- view.NewKeyDownEvent (new Key (KeyCode.N));
- Assert.True (keyPressInvoked);
- Assert.True (invokingKeyBindingsInvoked);
- Assert.False (processKeyPressInvoked);
- }
- [Theory]
- [InlineData (null, null)]
- [InlineData (true, true)]
- [InlineData (false, false)]
- public void OnInvokingKeyBindings_Returns_Nullable_Properly (bool? toReturn, bool? expected)
- {
- var view = new KeyBindingsTestView ();
- view.CommandReturns = toReturn;
- bool? result = view.OnInvokingKeyBindings (new Key (KeyCode.A));
- Assert.Equal (expected, result);
- }
- /// <summary>
- /// A view that overrides the OnKey* methods so we can test that they are called.
- /// </summary>
- public class KeyBindingsTestView : View {
- public bool? CommandReturns { get; set; }
- public KeyBindingsTestView ()
- {
- CanFocus = true;
- AddCommand (Command.Default, () => CommandReturns);
- KeyBindings.Add (KeyCode.A, Command.Default);
- }
- }
- [Fact]
- public void KeyDown_Handled_True_Stops_Processing ()
- {
- bool keyDown = false;
- bool invokingKeyBindings = false;
- bool keyPressed = false;
- var view = new OnKeyTestView ();
- Assert.True (view.CanFocus);
- view.CancelVirtualMethods = false;
- view.KeyDown += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyDown);
- Assert.False (view.OnKeyDownContinued);
- e.Handled = true;
- keyDown = true;
- };
- view.InvokingKeyBindings += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyPressed);
- Assert.False (view.OnInvokingKeyBindingsContinued);
- e.Handled = true;
- invokingKeyBindings = true;
- };
- view.ProcessKeyDown += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyPressed);
- Assert.False (view.OnKeyPressedContinued);
- e.Handled = true;
- keyPressed = true;
- };
- view.NewKeyDownEvent (new Key (KeyCode.A));
- Assert.True (keyDown);
- Assert.False (invokingKeyBindings);
- Assert.False (keyPressed);
- Assert.False (view.OnKeyDownContinued);
- Assert.False (view.OnInvokingKeyBindingsContinued);
- Assert.False (view.OnKeyPressedContinued);
- }
- [Fact]
- public void InvokingKeyBindings_Handled_True_Stops_Processing ()
- {
- bool keyDown = false;
- bool invokingKeyBindings = false;
- bool keyPressed = false;
- var view = new OnKeyTestView ();
- Assert.True (view.CanFocus);
- view.CancelVirtualMethods = false;
- view.KeyDown += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyDown);
- Assert.False (view.OnKeyDownContinued);
- e.Handled = false;
- keyDown = true;
- };
- view.InvokingKeyBindings += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyPressed);
- Assert.False (view.OnInvokingKeyBindingsContinued);
- e.Handled = true;
- invokingKeyBindings = true;
- };
- view.ProcessKeyDown += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyPressed);
- Assert.False (view.OnKeyPressedContinued);
- e.Handled = true;
- keyPressed = true;
- };
- view.NewKeyDownEvent (new Key (KeyCode.A));
- Assert.True (keyDown);
- Assert.True (invokingKeyBindings);
- Assert.False (keyPressed);
- Assert.True (view.OnKeyDownContinued);
- Assert.False (view.OnInvokingKeyBindingsContinued);
- Assert.False (view.OnKeyPressedContinued);
- }
- [Fact]
- public void KeyPressed_Handled_True_Stops_Processing ()
- {
- bool keyDown = false;
- bool invokingKeyBindings = false;
- bool keyPressed = false;
- var view = new OnKeyTestView ();
- Assert.True (view.CanFocus);
- view.CancelVirtualMethods = false;
- view.KeyDown += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyDown);
- Assert.False (view.OnKeyDownContinued);
- e.Handled = false;
- keyDown = true;
- };
- view.InvokingKeyBindings += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyPressed);
- Assert.False (view.OnInvokingKeyBindingsContinued);
- e.Handled = false;
- invokingKeyBindings = true;
- };
- view.ProcessKeyDown += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyPressed);
- Assert.False (view.OnKeyPressedContinued);
- e.Handled = true;
- keyPressed = true;
- };
- view.NewKeyDownEvent (new Key (KeyCode.A));
- Assert.True (keyDown);
- Assert.True (invokingKeyBindings);
- Assert.True (keyPressed);
- Assert.True (view.OnKeyDownContinued);
- Assert.True (view.OnInvokingKeyBindingsContinued);
- Assert.False (view.OnKeyPressedContinued);
- }
- [Fact]
- public void KeyUp_Handled_True_Stops_Processing ()
- {
- bool keyUp = false;
- var view = new OnKeyTestView ();
- Assert.True (view.CanFocus);
- view.CancelVirtualMethods = false;
- view.KeyUp += (s, e) => {
- Assert.Equal (KeyCode.A, e.KeyCode);
- Assert.False (keyUp);
- Assert.False (view.OnKeyPressedContinued);
- e.Handled = true;
- keyUp = true;
- };
- view.NewKeyUpEvent (new Key (KeyCode.A));
- Assert.True (keyUp);
- Assert.False (view.OnKeyUpContinued);
- Assert.False (view.OnKeyDownContinued);
- Assert.False (view.OnInvokingKeyBindingsContinued);
- Assert.False (view.OnKeyPressedContinued);
- }
- /// <summary>
- /// A view that overrides the OnKey* methods so we can test that they are called.
- /// </summary>
- public class OnKeyTestView : View {
- public bool CancelVirtualMethods { set; private get; }
- public OnKeyTestView () => CanFocus = true;
- public override string Text { get; set; }
- public bool OnKeyDownContinued { get; set; }
- public bool OnInvokingKeyBindingsContinued { get; set; }
- public bool OnKeyPressedContinued { get; set; }
- public bool OnKeyUpContinued { get; set; }
- public override bool OnKeyDown (Key keyEvent)
- {
- if (base.OnKeyDown (keyEvent)) {
- return true;
- }
- OnKeyDownContinued = true;
- return CancelVirtualMethods;
- }
- public override bool? OnInvokingKeyBindings (Key keyEvent)
- {
- bool? handled = base.OnInvokingKeyBindings (keyEvent);
- if (handled != null && (bool)handled) {
- return true;
- }
- OnInvokingKeyBindingsContinued = true;
- return CancelVirtualMethods;
- }
- public override bool OnProcessKeyDown (Key keyEvent)
- {
- if (base.OnProcessKeyDown (keyEvent)) {
- return true;
- }
- OnKeyPressedContinued = true;
- return CancelVirtualMethods;
- }
- public override bool OnKeyUp (Key keyEvent)
- {
- if (base.OnKeyUp (keyEvent)) {
- return true;
- }
- OnKeyUpContinued = true;
- return CancelVirtualMethods;
- }
- }
- [Theory]
- [InlineData (true, false, false)]
- [InlineData (true, true, false)]
- [InlineData (true, true, true)]
- public void Events_Are_Called_With_Only_Key_Modifiers (bool shift, bool alt, bool control)
- {
- bool keyDown = false;
- bool keyPressed = false;
- bool keyUp = false;
- var view = new OnKeyTestView ();
- view.CancelVirtualMethods = false;
- view.KeyDown += (s, e) => {
- Assert.Equal (KeyCode.Null, e.KeyCode & ~KeyCode.CtrlMask & ~KeyCode.AltMask & ~KeyCode.ShiftMask);
- Assert.Equal (shift, e.IsShift);
- Assert.Equal (alt, e.IsAlt);
- Assert.Equal (control, e.IsCtrl);
- Assert.False (keyDown);
- Assert.False (view.OnKeyDownContinued);
- keyDown = true;
- };
- view.ProcessKeyDown += (s, e) => {
- keyPressed = true;
- };
- view.KeyUp += (s, e) => {
- Assert.Equal (KeyCode.Null, e.KeyCode & ~KeyCode.CtrlMask & ~KeyCode.AltMask & ~KeyCode.ShiftMask);
- Assert.Equal (shift, e.IsShift);
- Assert.Equal (alt, e.IsAlt);
- Assert.Equal (control, e.IsCtrl);
- Assert.False (keyUp);
- Assert.False (view.OnKeyUpContinued);
- keyUp = true;
- };
- //view.ProcessKeyDownEvent (new (Key.Null | (shift ? Key.ShiftMask : 0) | (alt ? Key.AltMask : 0) | (control ? Key.CtrlMask : 0)));
- //Assert.True (keyDown);
- //Assert.True (view.OnKeyDownWasCalled);
- //Assert.True (view.OnProcessKeyDownWasCalled);
- view.NewKeyDownEvent (new Key (KeyCode.Null | (shift ? KeyCode.ShiftMask : 0) | (alt ? KeyCode.AltMask : 0) | (control ? KeyCode.CtrlMask : 0)));
- Assert.True (keyPressed);
- Assert.True (view.OnKeyDownContinued);
- Assert.True (view.OnKeyPressedContinued);
- view.NewKeyUpEvent (new Key (KeyCode.Null | (shift ? KeyCode.ShiftMask : 0) | (alt ? KeyCode.AltMask : 0) | (control ? KeyCode.CtrlMask : 0)));
- Assert.True (keyUp);
- Assert.True (view.OnKeyUpContinued);
- }
- /// <summary>
- /// This tests that when a new key down event is sent to the view
- /// the view will fire the 3 key-down related events: KeyDown, InvokingKeyBindings, and ProcessKeyDown.
- /// Note that KeyUp is independent.
- /// </summary>
- [Fact]
- public void AllViews_KeyDown_All_EventsFire ()
- {
- foreach (var view in TestHelpers.GetAllViews ()) {
- if (view == null) {
- _output.WriteLine ($"ERROR: null view from {nameof (TestHelpers.GetAllViews)}");
- continue;
- }
- _output.WriteLine ($"Testing {view.GetType ().Name}");
- bool keyDown = false;
- view.KeyDown += (s, a) => {
- a.Handled = false; // don't handle it so the other events are called
- keyDown = true;
- };
- bool invokingKeyBindings = false;
- view.InvokingKeyBindings += (s, a) => {
- a.Handled = false; // don't handle it so the other events are called
- invokingKeyBindings = true;
- };
- bool keyDownProcessed = false;
- view.ProcessKeyDown += (s, a) => {
- a.Handled = true;
- keyDownProcessed = true;
- };
- Assert.True (view.NewKeyDownEvent (Key.A)); // this will be true because the ProcessKeyDown event handled it
- Assert.True (keyDown);
- Assert.True (invokingKeyBindings);
- Assert.True (keyDownProcessed);
- view.Dispose ();
- }
- }
- /// <summary>
- /// This tests that when a new key up event is sent to the view
- /// the view will fire the 1 key-up related event: KeyUp
- /// </summary>
- [Fact]
- public void AllViews_KeyUp_All_EventsFire ()
- {
- foreach (var view in TestHelpers.GetAllViews ()) {
- if (view == null) {
- _output.WriteLine ($"ERROR: null view from {nameof (TestHelpers.GetAllViews)}");
- continue;
- }
- _output.WriteLine ($"Testing {view.GetType ().Name}");
- bool keyUp = false;
- view.KeyUp += (s, a) => {
- a.Handled = true;
- keyUp = true;
- };
- Assert.True (view.NewKeyUpEvent (Key.A)); // this will be true because the KeyUp event handled it
- Assert.True (keyUp);
- view.Dispose ();
- }
- }
- }
|