123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 |
- using Xunit.Abstractions;
- namespace Terminal.Gui.ViewMouseTests;
- [Trait ("Category", "Input")]
- public class MouseTests (ITestOutputHelper output) : TestsAllViews
- {
- [Theory]
- [InlineData (false, false, false)]
- [InlineData (true, false, true)]
- [InlineData (true, true, true)]
- public void MouseClick_SetsFocus_If_CanFocus (bool canFocus, bool setFocus, bool expectedHasFocus)
- {
- var superView = new View { CanFocus = true, Height = 1, Width = 15 };
- var focusedView = new View { CanFocus = true, Width = 1, Height = 1 };
- var testView = new View { CanFocus = canFocus, X = 4, Width = 4, Height = 1 };
- superView.Add (focusedView, testView);
- focusedView.SetFocus ();
- Assert.True (superView.HasFocus);
- Assert.True (focusedView.HasFocus);
- Assert.False (testView.HasFocus);
- if (setFocus)
- {
- testView.SetFocus ();
- }
- testView.NewMouseEvent (new () { Position = new (0, 0), Flags = MouseFlags.Button1Clicked });
- Assert.True (superView.HasFocus);
- Assert.Equal (expectedHasFocus, testView.HasFocus);
- }
- [Theory]
- [InlineData (false, false, 1)]
- [InlineData (true, false, 1)]
- [InlineData (true, true, 1)]
- public void MouseClick_Raises_Selecting (bool canFocus, bool setFocus, int expectedSelectingCount)
- {
- var superView = new View { CanFocus = true, Height = 1, Width = 15 };
- var focusedView = new View { CanFocus = true, Width = 1, Height = 1 };
- var testView = new View { CanFocus = canFocus, X = 4, Width = 4, Height = 1 };
- superView.Add (focusedView, testView);
- focusedView.SetFocus ();
- Assert.True (superView.HasFocus);
- Assert.True (focusedView.HasFocus);
- Assert.False (testView.HasFocus);
- if (setFocus)
- {
- testView.SetFocus ();
- }
- int selectingCount = 0;
- testView.Selecting += (sender, args) => selectingCount++;
- testView.NewMouseEvent (new () { Position = new (0, 0), Flags = MouseFlags.Button1Clicked });
- Assert.True (superView.HasFocus);
- Assert.Equal (expectedSelectingCount, selectingCount);
- }
- // TODO: Add more tests that ensure the above test works with positive adornments
- // Test drag to move
- [Theory]
- [InlineData (0, 0, 0, 0, false)]
- [InlineData (0, 0, 0, 4, false)]
- [InlineData (1, 0, 0, 4, false)]
- [InlineData (0, 1, 0, 4, true)]
- [InlineData (0, 0, 1, 4, false)]
- [InlineData (1, 1, 0, 3, false)]
- [InlineData (1, 1, 0, 4, false)]
- [InlineData (1, 1, 0, 5, true)]
- [InlineData (1, 1, 0, 6, false)]
- [InlineData (1, 1, 0, 11, false)]
- [InlineData (1, 1, 0, 12, true)]
- [InlineData (1, 1, 0, 13, false)]
- [InlineData (1, 1, 0, 14, false)]
- [AutoInitShutdown]
- public void ButtonPressed_In_Border_Starts_Drag (int marginThickness, int borderThickness, int paddingThickness, int xy, bool expectedMoved)
- {
- var testView = new View
- {
- CanFocus = true,
- X = 4,
- Y = 4,
- Width = 10,
- Height = 10,
- Arrangement = ViewArrangement.Movable
- };
- testView.Margin.Thickness = new (marginThickness);
- testView.Border.Thickness = new (borderThickness);
- testView.Padding.Thickness = new (paddingThickness);
- var top = new Toplevel ();
- top.Add (testView);
- Application.Begin (top);
- Assert.Equal (new Point (4, 4), testView.Frame.Location);
- Application.RaiseMouseEvent (new () { ScreenPosition = new (xy, xy), Flags = MouseFlags.Button1Pressed });
- Application.RaiseMouseEvent (new () { ScreenPosition = new (xy + 1, xy + 1), Flags = MouseFlags.Button1Pressed | MouseFlags.ReportMousePosition });
- Assert.Equal (expectedMoved, new Point (5, 5) == testView.Frame.Location);
- top.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.WheeledUp | MouseFlags.ButtonCtrl, MouseFlags.WheeledLeft)]
- [InlineData (MouseFlags.WheeledDown | MouseFlags.ButtonCtrl, MouseFlags.WheeledRight)]
- public void WheeledLeft_WheeledRight (MouseFlags mouseFlags, MouseFlags expectedMouseFlagsFromEvent)
- {
- MouseFlags mouseFlagsFromEvent = MouseFlags.None;
- var view = new View ();
- view.MouseEvent += (s, e) => mouseFlagsFromEvent = e.MouseEvent.Flags;
- view.NewMouseEvent (new MouseEvent () { Flags = mouseFlags });
- Assert.Equal (mouseFlagsFromEvent, expectedMouseFlagsFromEvent);
- }
- [Fact]
- public void NewMouseEvent_Invokes_MouseEvent_Properly ()
- {
- View view = new ()
- {
- Width = 1,
- Height = 1,
- };
- bool mouseEventInvoked = false;
- view.MouseEvent += (s, e) =>
- {
- mouseEventInvoked = true;
- e.Handled = true;
- };
- MouseEvent me = new ();
- view.NewMouseEvent (me);
- Assert.True (mouseEventInvoked);
- Assert.True (me.Handled);
- view.Dispose ();
- }
- [Theory]
- [MemberData (nameof (AllViewTypes))]
- public void AllViews_NewMouseEvent_Enabled_False_Does_Not_Set_Handled (Type viewType)
- {
- var view = CreateInstanceIfNotGeneric (viewType);
- if (view == null)
- {
- output.WriteLine ($"Ignoring {viewType} - It's a Generic");
- return;
- }
- view.Enabled = false;
- var me = new MouseEvent ();
- view.NewMouseEvent (me);
- Assert.False (me.Handled);
- view.Dispose ();
- }
- [Theory]
- [MemberData (nameof (AllViewTypes))]
- public void AllViews_NewMouseEvent_Clicked_Enabled_False_Does_Not_Set_Handled (Type viewType)
- {
- var view = CreateInstanceIfNotGeneric (viewType);
- if (view == null)
- {
- output.WriteLine ($"Ignoring {viewType} - It's a Generic");
- return;
- }
- view.Enabled = false;
- var me = new MouseEvent ()
- {
- Flags = MouseFlags.Button1Clicked
- };
- view.NewMouseEvent (me);
- Assert.False (me.Handled);
- view.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.Button1Pressed, MouseFlags.Button1Released, MouseFlags.Button1Clicked)]
- [InlineData (MouseFlags.Button2Pressed, MouseFlags.Button2Released, MouseFlags.Button2Clicked)]
- [InlineData (MouseFlags.Button3Pressed, MouseFlags.Button3Released, MouseFlags.Button3Clicked)]
- [InlineData (MouseFlags.Button4Pressed, MouseFlags.Button4Released, MouseFlags.Button4Clicked)]
- public void WantContinuousButtonPressed_False_Button_Press_Release_DoesNotClick (MouseFlags pressed, MouseFlags released, MouseFlags clicked)
- {
- var me = new MouseEvent ();
- var view = new View ()
- {
- Width = 1,
- Height = 1,
- WantContinuousButtonPressed = false
- };
- var clickedCount = 0;
- view.MouseClick += (s, e) => clickedCount++;
- me.Flags = pressed;
- view.NewMouseEvent (me);
- Assert.Equal (0, clickedCount);
- me.Handled = false;
- me.Flags = pressed;
- view.NewMouseEvent (me);
- Assert.Equal (0, clickedCount);
- me.Handled = false;
- me.Flags = released;
- view.NewMouseEvent (me);
- Assert.Equal (0, clickedCount);
- me.Handled = false;
- me.Flags = clicked;
- view.NewMouseEvent (me);
- Assert.Equal (1, clickedCount);
- view.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.Button1Clicked)]
- [InlineData (MouseFlags.Button2Clicked)]
- [InlineData (MouseFlags.Button3Clicked)]
- [InlineData (MouseFlags.Button4Clicked)]
- public void WantContinuousButtonPressed_True_Button_Clicked_Raises_MouseClick (MouseFlags clicked)
- {
- var me = new MouseEvent ();
- var view = new View ()
- {
- Width = 1,
- Height = 1,
- WantContinuousButtonPressed = true
- };
- var clickedCount = 0;
- view.MouseClick += (s, e) => clickedCount++;
- me.Flags = clicked;
- view.NewMouseEvent (me);
- Assert.Equal (1, clickedCount);
- view.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.Button1Clicked)]
- [InlineData (MouseFlags.Button2Clicked)]
- [InlineData (MouseFlags.Button3Clicked)]
- [InlineData (MouseFlags.Button4Clicked)]
- public void WantContinuousButtonPressed_True_Button_Clicked_Raises_Selecting (MouseFlags clicked)
- {
- var me = new MouseEvent ();
- var view = new View ()
- {
- Width = 1,
- Height = 1,
- WantContinuousButtonPressed = true
- };
- var selectingCount = 0;
- view.Selecting += (s, e) => selectingCount++;
- me.Flags = clicked;
- view.NewMouseEvent (me);
- Assert.Equal (1, selectingCount);
- view.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.Button1Pressed, MouseFlags.Button1Released)]
- [InlineData (MouseFlags.Button2Pressed, MouseFlags.Button2Released)]
- [InlineData (MouseFlags.Button3Pressed, MouseFlags.Button3Released)]
- [InlineData (MouseFlags.Button4Pressed, MouseFlags.Button4Released)]
- public void WantContinuousButtonPressed_True_And_WantMousePositionReports_True_Button_Press_Release_Clicks (MouseFlags pressed, MouseFlags released)
- {
- var me = new MouseEvent ();
- var view = new View ()
- {
- Width = 1,
- Height = 1,
- WantContinuousButtonPressed = true,
- WantMousePositionReports = true
- };
- var clickedCount = 0;
- view.MouseClick += (s, e) => clickedCount++;
- me.Flags = pressed;
- view.NewMouseEvent (me);
- Assert.Equal (1, clickedCount);
- me.Flags = released;
- view.NewMouseEvent (me);
- Assert.Equal (1, clickedCount);
- view.Dispose ();
- }
- [Theory]
- [InlineData (MouseFlags.Button1Pressed, MouseFlags.Button1Released, MouseFlags.Button1Clicked)]
- [InlineData (MouseFlags.Button2Pressed, MouseFlags.Button2Released, MouseFlags.Button2Clicked)]
- [InlineData (MouseFlags.Button3Pressed, MouseFlags.Button3Released, MouseFlags.Button3Clicked)]
- [InlineData (MouseFlags.Button4Pressed, MouseFlags.Button4Released, MouseFlags.Button4Clicked)]
- public void WantContinuousButtonPressed_True_And_WantMousePositionReports_True_Button_Press_Release_Clicks_Repeatedly (MouseFlags pressed, MouseFlags released, MouseFlags clicked)
- {
- var me = new MouseEvent ();
- var view = new View ()
- {
- Width = 1,
- Height = 1,
- WantContinuousButtonPressed = true,
- WantMousePositionReports = true
- };
- var clickedCount = 0;
- view.MouseClick += (s, e) => clickedCount++;
- me.Flags = pressed;
- view.NewMouseEvent (me);
- Assert.Equal (1, clickedCount);
- me.Handled = false;
- me.Flags = pressed;
- view.NewMouseEvent (me);
- Assert.Equal (2, clickedCount);
- me.Handled = false;
- me.Flags = released;
- view.NewMouseEvent (me);
- Assert.Equal (2, clickedCount);
- me.Handled = false;
- me.Flags = clicked;
- view.NewMouseEvent (me);
- Assert.Equal (2, clickedCount);
- view.Dispose ();
- Application.ResetState (ignoreDisposed: true);
- }
- [Fact]
- public void WantContinuousButtonPressed_True_And_WantMousePositionReports_True_Move_InViewport_OutOfViewport_Keeps_Counting ()
- {
- var me = new MouseEvent ();
- var view = new View ()
- {
- Width = 1,
- Height = 1,
- WantContinuousButtonPressed = true,
- WantMousePositionReports = true
- };
- var clickedCount = 0;
- view.MouseClick += (s, e) => clickedCount++;
- // Start in Viewport
- me.Flags = MouseFlags.Button1Pressed;
- me.Position = me.Position with { X = 0 };
- view.NewMouseEvent (me);
- Assert.Equal (1, clickedCount);
- me.Handled = false;
- // Move out of Viewport
- me.Flags = MouseFlags.Button1Pressed;
- me.Position = me.Position with { X = 1 };
- view.NewMouseEvent (me);
- Assert.Equal (2, clickedCount);
- me.Handled = false;
- // Move into Viewport
- me.Flags = MouseFlags.Button1Pressed;
- me.Position = me.Position with { X = 0 };
- view.NewMouseEvent (me);
- Assert.Equal (3, clickedCount);
- me.Handled = false;
- view.Dispose ();
- Application.ResetState (ignoreDisposed: true);
- }
- [Theory]
- [InlineData (HighlightStyle.None, 0, 0)]
- [InlineData (HighlightStyle.Pressed | HighlightStyle.PressedOutside, 1, 1)]
- public void HighlightOnPress_Fires_Events_And_Highlights (HighlightStyle highlightOnPress, int expectedEnabling, int expectedDisabling)
- {
- var view = new View ()
- {
- CanFocus = true,
- HighlightStyle = highlightOnPress,
- Height = 1,
- Width = 1
- };
- int enablingHighlight = 0;
- int disablingHighlight = 0;
- view.Highlight += View_Highlight;
- view.ColorScheme = new ColorScheme (new Attribute (ColorName16.Red, ColorName16.Blue));
- ColorScheme originalColorScheme = view.ColorScheme;
- view.NewMouseEvent (new () { Flags = MouseFlags.Button1Pressed, });
- if (highlightOnPress != HighlightStyle.None)
- {
- Assert.NotEqual (originalColorScheme, view.ColorScheme);
- }
- else
- {
- Assert.Equal (originalColorScheme, view.ColorScheme);
- }
- view.NewMouseEvent (new () { Flags = MouseFlags.Button1Released, });
- Assert.Equal (originalColorScheme, view.ColorScheme);
- Assert.Equal (expectedEnabling, enablingHighlight);
- Assert.Equal (expectedDisabling, disablingHighlight);
- view.Dispose ();
- return;
- void View_Highlight (object sender, CancelEventArgs<HighlightStyle> e)
- {
- if (e.NewValue == HighlightStyle.None)
- {
- disablingHighlight++;
- }
- else
- {
- enablingHighlight++;
- }
- }
- }
- // TODO: Add tests for each combination of HighlightFlags
- [Theory]
- [InlineData (0)]
- [InlineData (1)]
- [InlineData (10)]
- public void HighlightOnPress_Move_Keeps_Highlight (int x)
- {
- var view = new View ()
- {
- CanFocus = true,
- HighlightStyle = HighlightStyle.Pressed | HighlightStyle.PressedOutside,
- Height = 1,
- Width = 1
- };
- int enablingHighlight = 0;
- int disablingHighlight = 0;
- view.Highlight += View_Highlight;
- bool inViewport = view.Viewport.Contains (x, 0);
- // Start at 0,0 ; in viewport
- view.NewMouseEvent (new () { Flags = MouseFlags.Button1Pressed });
- Assert.Equal (1, enablingHighlight);
- Assert.Equal (0, disablingHighlight);
- // Move to x,0
- view.NewMouseEvent (new () { Flags = MouseFlags.Button1Pressed });
- if (inViewport)
- {
- Assert.Equal (2, enablingHighlight);
- Assert.Equal (0, disablingHighlight);
- }
- else
- {
- Assert.Equal (2, enablingHighlight);
- Assert.Equal (0, disablingHighlight);
- }
- // Move backto 0,0 ; in viewport
- view.NewMouseEvent (new () { Flags = MouseFlags.Button1Pressed });
- if (inViewport)
- {
- Assert.Equal (3, enablingHighlight);
- Assert.Equal (0, disablingHighlight);
- }
- else
- {
- Assert.Equal (3, enablingHighlight);
- Assert.Equal (0, disablingHighlight);
- }
- view.Dispose ();
- return;
- void View_Highlight (object sender, CancelEventArgs<HighlightStyle> e)
- {
- if (e.NewValue == HighlightStyle.None)
- {
- disablingHighlight++;
- }
- else
- {
- enablingHighlight++;
- }
- }
- }
- }
|