123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671 |
- using System.Collections.ObjectModel;
- using System.Diagnostics;
- using System.Reflection;
- using Terminal.Gui;
- using UICatalog;
- using UnitTests;
- using Xunit.Abstractions;
- namespace IntegrationTests.UICatalog;
- public class ScenarioTests : TestsAllViews
- {
- public ScenarioTests (ITestOutputHelper output)
- {
- #if DEBUG_IDISPOSABLE
- View.DebugIDisposable = true;
- View.Instances.Clear ();
- #endif
- _output = output;
- }
- private readonly ITestOutputHelper _output;
- private object? _timeoutLock;
- /// <summary>
- /// <para>This runs through all Scenarios defined in UI Catalog, calling Init, Setup, and Run.</para>
- /// <para>Should find any Scenarios which crash on load or do not respond to <see cref="Application.RequestStop()"/>.</para>
- /// </summary>
- [Theory]
- [MemberData (nameof (AllScenarioTypes))]
- public void All_Scenarios_Quit_And_Init_Shutdown_Properly (Type scenarioType)
- {
- Assert.Null (_timeoutLock);
- _timeoutLock = new ();
- // Disable any UIConfig settings
- ConfigLocations savedConfigLocations = ConfigurationManager.Locations;
- ConfigurationManager.Locations = ConfigLocations.Default;
- // If a previous test failed, this will ensure that the Application is in a clean state
- Application.ResetState (true);
- _output.WriteLine ($"Running Scenario '{scenarioType}'");
- var scenario = Activator.CreateInstance (scenarioType) as Scenario;
- uint abortTime = 1500;
- object? timeout = null;
- var initialized = false;
- var shutdown = false;
- var iterationCount = 0;
- Application.InitializedChanged += OnApplicationOnInitializedChanged;
- Application.ForceDriver = "FakeDriver";
- scenario!.Main ();
- scenario.Dispose ();
- scenario = null;
- Application.ForceDriver = string.Empty;
- Application.InitializedChanged -= OnApplicationOnInitializedChanged;
- lock (_timeoutLock)
- {
- if (timeout is { })
- {
- timeout = null;
- }
- }
- Assert.True (initialized);
- Assert.True (shutdown);
- #if DEBUG_IDISPOSABLE
- Assert.Empty (View.Instances);
- #endif
- lock (_timeoutLock)
- {
- _timeoutLock = null;
- }
- // Restore the configuration locations
- ConfigurationManager.Locations = savedConfigLocations;
- ConfigurationManager.Reset ();
- return;
- void OnApplicationOnInitializedChanged (object? s, EventArgs<bool> a)
- {
- if (a.CurrentValue)
- {
- Application.Iteration += OnApplicationOnIteration;
- initialized = true;
- lock (_timeoutLock)
- {
- timeout = Application.AddTimeout (TimeSpan.FromMilliseconds (abortTime), ForceCloseCallback);
- }
- }
- else
- {
- Application.Iteration -= OnApplicationOnIteration;
- shutdown = true;
- }
- _output.WriteLine ($"Initialized == {a.CurrentValue}");
- }
- // If the scenario doesn't close within 500ms, this will force it to quit
- bool ForceCloseCallback ()
- {
- lock (_timeoutLock)
- {
- if (timeout is { })
- {
- timeout = null;
- }
- }
- Assert.Fail (
- $"Scenario Failed to Quit with {Application.QuitKey} after {abortTime}ms and {iterationCount} iterations. Force quit.");
- // Restore the configuration locations
- ConfigurationManager.Locations = savedConfigLocations;
- ConfigurationManager.Reset ();
- Application.ResetState (true);
- return false;
- }
- void OnApplicationOnIteration (object? s, IterationEventArgs a)
- {
- iterationCount++;
- if (Application.Initialized)
- {
- // Press QuitKey
- _output.WriteLine ($"Attempting to quit with {Application.QuitKey}");
- Application.RaiseKeyDownEvent (Application.QuitKey);
- }
- }
- }
- public static IEnumerable<object []> AllScenarioTypes =>
- typeof (Scenario).Assembly
- .GetTypes ()
- .Where (type => type.IsClass && !type.IsAbstract && type.IsSubclassOf (typeof (Scenario)))
- .Select (type => new object [] { type });
- [Fact]
- public void Run_All_Views_Tester_Scenario ()
- {
- // Disable any UIConfig settings
- ConfigLocations savedConfigLocations = ConfigurationManager.Locations;
- ConfigurationManager.Locations = ConfigLocations.Default;
- View? curView = null;
- // Settings
- var xVal = 0;
- var yVal = 0;
- var wVal = 0;
- var hVal = 0;
- List<string> posNames = ["Percent", "AnchorEnd", "Center", "Absolute"];
- List<string> dimNames = ["Auto", "Percent", "Fill", "Absolute"];
- Application.Init (new FakeDriver ());
- var top = new Toplevel ();
- Dictionary<string, Type> viewClasses = GetAllViewClasses().ToDictionary (t => t.Name);
- Window leftPane = new ()
- {
- Title = "Classes",
- X = 0,
- Y = 0,
- Width = 15,
- Height = Dim.Fill (1), // for status bar
- CanFocus = false,
- ColorScheme = Colors.ColorSchemes ["TopLevel"]
- };
- ListView classListView = new ()
- {
- X = 0,
- Y = 0,
- Width = Dim.Fill (),
- Height = Dim.Fill (),
- AllowsMarking = false,
- ColorScheme = Colors.ColorSchemes ["TopLevel"],
- Source = new ListWrapper<string> (new (viewClasses.Keys.ToList ()))
- };
- leftPane.Add (classListView);
- FrameView settingsPane = new ()
- {
- X = Pos.Right (leftPane),
- Y = 0, // for menu
- Width = Dim.Fill (),
- Height = 10,
- CanFocus = false,
- ColorScheme = Colors.ColorSchemes ["TopLevel"],
- Title = "Settings"
- };
- var radioItems = new [] { "Percent(x)", "AnchorEnd(x)", "Center", "Absolute(x)" };
- FrameView locationFrame = new ()
- {
- X = 0,
- Y = 0,
- Height = 3 + radioItems.Length,
- Width = 36,
- Title = "Location (Pos)"
- };
- settingsPane.Add (locationFrame);
- var label = new Label { X = 0, Y = 0, Text = "x:" };
- locationFrame.Add (label);
- RadioGroup xRadioGroup = new () { X = 0, Y = Pos.Bottom (label), RadioLabels = radioItems };
- TextField xText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{xVal}" };
- locationFrame.Add (xText);
- locationFrame.Add (xRadioGroup);
- radioItems = new [] { "Percent(y)", "AnchorEnd(y)", "Center", "Absolute(y)" };
- label = new () { X = Pos.Right (xRadioGroup) + 1, Y = 0, Text = "y:" };
- locationFrame.Add (label);
- TextField yText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{yVal}" };
- locationFrame.Add (yText);
- RadioGroup yRadioGroup = new () { X = Pos.X (label), Y = Pos.Bottom (label), RadioLabels = radioItems };
- locationFrame.Add (yRadioGroup);
- FrameView sizeFrame = new ()
- {
- X = Pos.Right (locationFrame),
- Y = Pos.Y (locationFrame),
- Height = 3 + radioItems.Length,
- Width = 40,
- Title = "Size (Dim)"
- };
- radioItems = new [] { "Auto()", "Percent(width)", "Fill(width)", "Absolute(width)" };
- label = new () { X = 0, Y = 0, Text = "width:" };
- sizeFrame.Add (label);
- RadioGroup wRadioGroup = new () { X = 0, Y = Pos.Bottom (label), RadioLabels = radioItems };
- TextField wText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{wVal}" };
- sizeFrame.Add (wText);
- sizeFrame.Add (wRadioGroup);
- radioItems = new [] { "Auto()", "Percent(height)", "Fill(height)", "Absolute(height)" };
- label = new () { X = Pos.Right (wRadioGroup) + 1, Y = 0, Text = "height:" };
- sizeFrame.Add (label);
- TextField hText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{hVal}" };
- sizeFrame.Add (hText);
- RadioGroup hRadioGroup = new () { X = Pos.X (label), Y = Pos.Bottom (label), RadioLabels = radioItems };
- sizeFrame.Add (hRadioGroup);
- settingsPane.Add (sizeFrame);
- FrameView hostPane = new ()
- {
- X = Pos.Right (leftPane),
- Y = Pos.Bottom (settingsPane),
- Width = Dim.Fill (),
- Height = Dim.Fill (1), // + 1 for status bar
- ColorScheme = Colors.ColorSchemes ["Dialog"]
- };
- classListView.OpenSelectedItem += (s, a) => { settingsPane.SetFocus (); };
- classListView.SelectedItemChanged += (s, args) =>
- {
- // Remove existing class, if any
- if (curView is {})
- {
- curView.SubViewsLaidOut -= LayoutCompleteHandler;
- hostPane.Remove (curView);
- curView.Dispose ();
- curView = null;
- hostPane.FillRect (hostPane.Viewport);
- }
- curView = CreateClass (viewClasses.Values.ToArray () [classListView.SelectedItem]);
- };
- xRadioGroup.SelectedItemChanged += (s, selected) => DimPosChanged (curView);
- xText.TextChanged += (s, args) =>
- {
- try
- {
- xVal = int.Parse (xText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- yText.TextChanged += (s, e) =>
- {
- try
- {
- yVal = int.Parse (yText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- yRadioGroup.SelectedItemChanged += (s, selected) => DimPosChanged (curView);
- wRadioGroup.SelectedItemChanged += (s, selected) => DimPosChanged (curView);
- wText.TextChanged += (s, args) =>
- {
- try
- {
- wVal = int.Parse (wText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- hText.TextChanged += (s, args) =>
- {
- try
- {
- hVal = int.Parse (hText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- hRadioGroup.SelectedItemChanged += (s, selected) => DimPosChanged (curView);
- top.Add (leftPane, settingsPane, hostPane);
- top.LayoutSubViews ();
- curView = CreateClass (viewClasses.First ().Value);
- var iterations = 0;
- Application.Iteration += (s, a) =>
- {
- iterations++;
- if (iterations < viewClasses.Count)
- {
- classListView.MoveDown ();
- Assert.Equal (
- curView!.GetType ().Name,
- viewClasses.Values.ToArray () [classListView.SelectedItem].Name
- );
- }
- else
- {
- Application.RequestStop ();
- }
- };
- Application.Run (top);
- Assert.Equal (viewClasses.Count, iterations);
- top.Dispose ();
- Application.Shutdown ();
- // Restore the configuration locations
- ConfigurationManager.Locations = savedConfigLocations;
- ConfigurationManager.Reset ();
- void DimPosChanged (View? view)
- {
- if (view == null)
- {
- return;
- }
- try
- {
- switch (xRadioGroup.SelectedItem)
- {
- case 0:
- view.X = Pos.Percent (xVal);
- break;
- case 1:
- view.X = Pos.AnchorEnd (xVal);
- break;
- case 2:
- view.X = Pos.Center ();
- break;
- case 3:
- view.X = Pos.Absolute (xVal);
- break;
- }
- switch (yRadioGroup.SelectedItem)
- {
- case 0:
- view.Y = Pos.Percent (yVal);
- break;
- case 1:
- view.Y = Pos.AnchorEnd (yVal);
- break;
- case 2:
- view.Y = Pos.Center ();
- break;
- case 3:
- view.Y = Pos.Absolute (yVal);
- break;
- }
- switch (wRadioGroup.SelectedItem)
- {
- case 0:
- view.Width = Dim.Percent (wVal);
- break;
- case 1:
- view.Width = Dim.Fill (wVal);
- break;
- case 2:
- view.Width = Dim.Absolute (wVal);
- break;
- }
- switch (hRadioGroup.SelectedItem)
- {
- case 0:
- view.Height = Dim.Percent (hVal);
- break;
- case 1:
- view.Height = Dim.Fill (hVal);
- break;
- case 2:
- view.Height = Dim.Absolute (hVal);
- break;
- }
- }
- catch (Exception e)
- {
- MessageBox.ErrorQuery ("Exception", e.Message, "Ok");
- }
- UpdateTitle (view);
- }
- void UpdateSettings (View view)
- {
- var x = view.X.ToString ();
- var y = view.Y.ToString ();
- try
- {
- xRadioGroup.SelectedItem = posNames.IndexOf (posNames.First (s => x.Contains (s)));
- yRadioGroup.SelectedItem = posNames.IndexOf (posNames.First (s => y.Contains (s)));
- }
- catch (InvalidOperationException e)
- {
- // This is a hack to work around the fact that the Pos enum doesn't have an "Align" value yet
- Debug.WriteLine ($"{e}");
- }
- xText.Text = $"{view.Frame.X}";
- yText.Text = $"{view.Frame.Y}";
- var w = view.Width!.ToString ();
- var h = view.Height!.ToString ();
- wRadioGroup.SelectedItem = dimNames.IndexOf (dimNames.First (s => w.Contains (s)));
- hRadioGroup.SelectedItem = dimNames.IndexOf (dimNames.First (s => h.Contains (s)));
- wText.Text = $"{view.Frame.Width}";
- hText.Text = $"{view.Frame.Height}";
- }
- void UpdateTitle (View? view) { hostPane.Title = $"{view!.GetType ().Name} - {view.X}, {view.Y}, {view.Width}, {view.Height}"; }
- View? CreateClass (Type type)
- {
- // If we are to create a generic Type
- if (type.IsGenericType)
- {
- // For each of the <T> arguments
- List<Type> typeArguments = new ();
- // use <object>
- foreach (Type arg in type.GetGenericArguments ())
- {
- typeArguments.Add (typeof (object));
- }
- // And change what type we are instantiating from MyClass<T> to MyClass<object>
- type = type.MakeGenericType (typeArguments.ToArray ());
- }
- // Instantiate view
- var view = Activator.CreateInstance (type) as View;
- if (view is null)
- {
- return null;
- }
- if (view.Width is not DimAuto)
- {
- view.Width = Dim.Percent (75);
- }
- if (view.Height is not DimAuto)
- {
- view.Height = Dim.Percent (75);
- }
- // Set the colorscheme to make it stand out if is null by default
- view.ColorScheme ??= Colors.ColorSchemes ["Base"];
- // If the view supports a Text property, set it so we have something to look at
- if (view.GetType ().GetProperty ("Text") != null)
- {
- try
- {
- view.GetType ().GetProperty ("Text")?.GetSetMethod ()?.Invoke (view, new [] { "Test Text" });
- }
- catch (TargetInvocationException e)
- {
- MessageBox.ErrorQuery ("Exception", e.InnerException!.Message, "Ok");
- view = null;
- }
- }
- // If the view supports a Title property, set it so we have something to look at
- if (view != null && view.GetType ().GetProperty ("Title") != null)
- {
- if (view.GetType ().GetProperty ("Title")!.PropertyType == typeof (string))
- {
- view?.GetType ().GetProperty ("Title")?.GetSetMethod ()?.Invoke (view, new [] { "Test Title" });
- }
- else
- {
- view?.GetType ().GetProperty ("Title")?.GetSetMethod ()?.Invoke (view, new [] { "Test Title" });
- }
- }
- // If the view supports a Source property, set it so we have something to look at
- if (view != null
- && view.GetType ().GetProperty ("Source") != null
- && view.GetType ().GetProperty ("Source")!.PropertyType == typeof (IListDataSource))
- {
- ListWrapper<string> source = new (["Test Text #1", "Test Text #2", "Test Text #3"]);
- view?.GetType ().GetProperty ("Source")?.GetSetMethod ()?.Invoke (view, new [] { source });
- }
- // Add
- hostPane.Add (view);
- //DimPosChanged ();
- hostPane.LayoutSubViews ();
- hostPane.ClearViewport ();
- hostPane.SetNeedsDraw ();
- UpdateSettings (view!);
- UpdateTitle (view);
- view!.SubViewsLaidOut += LayoutCompleteHandler;
- return view;
- }
- void LayoutCompleteHandler (object? sender, LayoutEventArgs args) { UpdateTitle (curView); }
- }
- [Fact]
- public void Run_Generic ()
- {
- // Disable any UIConfig settings
- ConfigLocations savedConfigLocations = ConfigurationManager.Locations;
- ConfigurationManager.Locations = ConfigLocations.Default;
- ObservableCollection<Scenario> scenarios = Scenario.GetScenarios ();
- Assert.NotEmpty (scenarios);
- int item = scenarios.IndexOf (s => s.GetName ().Equals ("Generic", StringComparison.OrdinalIgnoreCase));
- Scenario generic = scenarios [item];
- Application.Init (new FakeDriver ());
- // BUGBUG: (#2474) For some reason ReadKey is not returning the QuitKey for some Scenarios
- // by adding this Space it seems to work.
- FakeConsole.PushMockKeyPress ((KeyCode)Application.QuitKey);
- var ms = 100;
- var abortCount = 0;
- Func<bool> abortCallback = () =>
- {
- abortCount++;
- _output.WriteLine ($"'Generic' abortCount {abortCount}");
- Application.RequestStop ();
- return false;
- };
- var iterations = 0;
- object? token = null;
- Application.Iteration += (s, a) =>
- {
- if (token == null)
- {
- // Timeout only must start at first iteration
- token = Application.AddTimeout (TimeSpan.FromMilliseconds (ms), abortCallback);
- }
- iterations++;
- _output.WriteLine ($"'Generic' iteration {iterations}");
- // Stop if we run out of control...
- if (iterations == 10)
- {
- _output.WriteLine ("'Generic' had to be force quit!");
- Application.RequestStop ();
- }
- };
- Application.KeyDown += (sender, args) => { Assert.Equal (Application.QuitKey, args.KeyCode); };
- generic.Main ();
- Assert.Equal (0, abortCount);
- // # of key up events should match # of iterations
- Assert.Equal (1, iterations);
- generic.Dispose ();
- // Shutdown must be called to safely clean up Application if Init has been called
- Application.Shutdown ();
- // Restore the configuration locations
- ConfigurationManager.Locations = savedConfigLocations;
- ConfigurationManager.Reset ();
- #if DEBUG_IDISPOSABLE
- Assert.Empty (View.Instances);
- #endif
- }
- }
|