| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490 |
- using Terminal.Gui.App;
- using Xunit.Abstractions;
- namespace ApplicationTests.Mouse;
- /// <summary>
- /// Parallelizable tests for mouse event routing and coordinate transformation.
- /// These tests validate mouse event handling without Application.Begin or global state.
- /// </summary>
- [Trait ("Category", "Input")]
- public class MouseEventRoutingTests (ITestOutputHelper output)
- {
- private readonly ITestOutputHelper _output = output;
- #region Mouse Event Routing to Views
- [Theory]
- [InlineData (5, 5, 5, 5, true)] // Click inside view
- [InlineData (0, 0, 0, 0, true)] // Click at origin
- [InlineData (9, 9, 9, 9, true)] // Click at far corner (view is 10x10)
- [InlineData (10, 10, -1, -1, false)] // Click outside view
- [InlineData (-1, -1, -1, -1, false)] // Click outside view
- public void View_NewMouseEvent_ReceivesCorrectCoordinates (int screenX, int screenY, int expectedViewX, int expectedViewY, bool shouldReceive)
- {
- // Arrange
- View view = new ()
- {
- X = 0,
- Y = 0,
- Width = 10,
- Height = 10
- };
- Point? receivedPosition = null;
- bool eventReceived = false;
- view.MouseEvent += (_, args) =>
- {
- eventReceived = true;
- receivedPosition = args.Position;
- };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new Point (screenX, screenY),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- if (shouldReceive)
- {
- Assert.True (eventReceived);
- Assert.NotNull (receivedPosition);
- Assert.Equal (expectedViewX, receivedPosition.Value.X);
- Assert.Equal (expectedViewY, receivedPosition.Value.Y);
- }
- view.Dispose ();
- }
- [Theory]
- [InlineData (0, 0, 5, 5, 5, 5, true)] // View at origin, click at (5,5) in view
- [InlineData (10, 10, 5, 5, 5, 5, true)] // View offset, but we still pass view-relative coords
- [InlineData (0, 0, 0, 0, 0, 0, true)] // View at origin, click at origin
- [InlineData (5, 5, 9, 9, 9, 9, true)] // View offset, click at far corner (view-relative)
- [InlineData (0, 0, 10, 10, -1, -1, false)] // Click outside view bounds
- [InlineData (0, 0, -1, -1, -1, -1, false)] // Click outside view bounds
- public void View_WithOffset_ReceivesCorrectCoordinates (
- int viewX,
- int viewY,
- int viewRelativeX,
- int viewRelativeY,
- int expectedViewX,
- int expectedViewY,
- bool shouldReceive)
- {
- // Arrange
- // Note: When testing View.NewMouseEvent directly (without Application routing),
- // coordinates are already view-relative. The view's X/Y position doesn't affect
- // the coordinate transformation at this level.
- View view = new ()
- {
- X = viewX,
- Y = viewY,
- Width = 10,
- Height = 10
- };
- Point? receivedPosition = null;
- bool eventReceived = false;
- view.MouseEvent += (_, args) =>
- {
- eventReceived = true;
- receivedPosition = args.Position;
- };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (viewRelativeX, viewRelativeY),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- if (shouldReceive)
- {
- Assert.True (eventReceived, $"Event should be received at view-relative ({viewRelativeX},{viewRelativeY})");
- Assert.NotNull (receivedPosition);
- Assert.Equal (expectedViewX, receivedPosition.Value.X);
- Assert.Equal (expectedViewY, receivedPosition.Value.Y);
- }
- view.Dispose ();
- }
- #endregion
- #region View Hierarchy Mouse Event Routing
- [Fact]
- public void SubView_ReceivesMouseEvent_WithCorrectRelativeCoordinates ()
- {
- // Arrange
- View superView = new ()
- {
- X = 0,
- Y = 0,
- Width = 20,
- Height = 20
- };
- View subView = new ()
- {
- X = 5,
- Y = 5,
- Width = 10,
- Height = 10
- };
- superView.Add (subView);
- Point? subViewReceivedPosition = null;
- bool subViewEventReceived = false;
- subView.MouseEvent += (_, args) =>
- {
- subViewEventReceived = true;
- subViewReceivedPosition = args.Position;
- };
- // Click at position (2, 2) relative to subView (which is at 5,5 relative to superView)
- MouseEventArgs mouseEvent = new ()
- {
- Position = new Point (2, 2), // Relative to subView
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- subView.NewMouseEvent (mouseEvent);
- // Assert
- Assert.True (subViewEventReceived);
- Assert.NotNull (subViewReceivedPosition);
- Assert.Equal (2, subViewReceivedPosition.Value.X);
- Assert.Equal (2, subViewReceivedPosition.Value.Y);
- subView.Dispose ();
- superView.Dispose ();
- }
- [Fact]
- public void MouseClick_OnSubView_RaisesSelectingEvent ()
- {
- // Arrange
- View superView = new ()
- {
- Width = 20,
- Height = 20
- };
- View subView = new ()
- {
- X = 5,
- Y = 5,
- Width = 10,
- Height = 10
- };
- superView.Add (subView);
- int activatingCount = 0;
- subView.Activating += (_, _) => activatingCount++;
- MouseEventArgs mouseEvent = new ()
- {
- Position = new Point (5, 5),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- subView.NewMouseEvent (mouseEvent);
- // Assert
- Assert.Equal (1, activatingCount);
- subView.Dispose ();
- superView.Dispose ();
- }
- #endregion
- #region Mouse Event Propagation
- [Fact]
- public void View_HandledEvent_StopsPropagation ()
- {
- // Arrange
- View view = new () { Width = 10, Height = 10 };
- bool handlerCalled = false;
- bool clickHandlerCalled = false;
- view.MouseEvent += (_, args) =>
- {
- handlerCalled = true;
- args.Handled = true; // Mark as handled
- };
- view.MouseEvent += (_, e) => { clickHandlerCalled = !e.IsSingleDoubleOrTripleClicked; ; };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- bool? result = view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.True (result.HasValue && result.Value); // Event was handled
- Assert.True (handlerCalled);
- Assert.False (clickHandlerCalled); // Click handler should not be called when event is handled
- view.Dispose ();
- }
- [Fact]
- public void View_UnhandledEvent_ContinuesProcessing ()
- {
- // Arrange
- View view = new () { Width = 10, Height = 10 };
- bool eventHandlerCalled = false;
- view.MouseEvent += (_, _) =>
- {
- eventHandlerCalled = true;
- // Don't set Handled = true
- };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.True (eventHandlerCalled);
- view.Dispose ();
- }
- #endregion
- #region Mouse Button Events
- [Theory]
- [InlineData (MouseFlags.Button1Pressed, 1, 0)]
- [InlineData (MouseFlags.Button1Released, 0, 1)]
- [InlineData (MouseFlags.Button1Clicked, 0, 0)]
- public void View_MouseButtonEvents_RaiseCorrectHandlers (MouseFlags flags, int expectedPressed, int expectedReleased)
- {
- // Arrange
- View view = new () { Width = 10, Height = 10 };
- int pressedCount = 0;
- int releasedCount = 0;
- view.MouseEvent += (_, args) =>
- {
- if (args.Flags.HasFlag (MouseFlags.Button1Pressed))
- {
- pressedCount++;
- }
- if (args.Flags.HasFlag (MouseFlags.Button1Released))
- {
- releasedCount++;
- }
- };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = flags
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.Equal (expectedPressed, pressedCount);
- Assert.Equal (expectedReleased, releasedCount);
- view.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.Button1Clicked)]
- [InlineData (MouseFlags.Button2Clicked)]
- [InlineData (MouseFlags.Button3Clicked)]
- [InlineData (MouseFlags.Button4Clicked)]
- public void View_AllMouseButtons_TriggerClickEvent (MouseFlags clickFlag)
- {
- // Arrange
- View view = new () { Width = 10, Height = 10 };
- int clickCount = 0;
- view.MouseEvent += (_, a) => clickCount += a.IsSingleDoubleOrTripleClicked ? 1 : 0;
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = clickFlag
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.Equal (1, clickCount);
- view.Dispose ();
- }
- #endregion
- #region Disabled View Tests
- [Fact]
- public void View_Disabled_DoesNotRaiseMouseEvent ()
- {
- // Arrange
- View view = new ()
- {
- Width = 10,
- Height = 10,
- Enabled = false
- };
- bool eventCalled = false;
- view.MouseEvent += (_, _) => { eventCalled = true; };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.False (eventCalled);
- view.Dispose ();
- }
- [Fact]
- public void View_Disabled_DoesNotRaiseSelectingEvent ()
- {
- // Arrange
- View view = new ()
- {
- Width = 10,
- Height = 10,
- Enabled = false
- };
- bool selectingCalled = false;
- view.Activating += (_, _) => { selectingCalled = true; };
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.False (selectingCalled);
- view.Dispose ();
- }
- #endregion
- #region Focus and Selection Tests
- [Theory]
- [InlineData (true, true)]
- [InlineData (false, false)]
- public void MouseClick_SetsFocus_BasedOnCanFocus (bool canFocus, bool expectFocus)
- {
- // Arrange
- View superView = new () { CanFocus = true, Width = 20, Height = 20 };
- View subView = new ()
- {
- X = 5,
- Y = 5,
- Width = 10,
- Height = 10,
- CanFocus = canFocus
- };
- superView.Add (subView);
- superView.SetFocus (); // Give superView focus first
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (2, 2),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- subView.NewMouseEvent (mouseEvent);
- // Assert
- Assert.Equal (expectFocus, subView.HasFocus);
- subView.Dispose ();
- superView.Dispose ();
- }
- [Fact]
- public void MouseClick_RaisesSelecting_WhenCanFocus ()
- {
- // Arrange
- View superView = new () { CanFocus = true, Width = 20, Height = 20 };
- View view = new ()
- {
- X = 5,
- Y = 5,
- Width = 10,
- Height = 10,
- CanFocus = true
- };
- superView.Add (view);
- int activatingCount = 0;
- view.Activating += (_, _) => activatingCount++;
- MouseEventArgs mouseEvent = new ()
- {
- Position = new (5, 5),
- Flags = MouseFlags.Button1Clicked
- };
- // Act
- view.NewMouseEvent (mouseEvent);
- // Assert
- Assert.Equal (1, activatingCount);
- view.Dispose ();
- superView.Dispose ();
- }
- #endregion
- }
|