i'm bad at git, large commit with lots of changes

main
resneptacle 4 weeks ago
parent 04677ab97a
commit a6377d1a57

@ -187,81 +187,3 @@ public class MessageBox {
return index; return index;
} }
} }
public class ColorPairs {
public static int Ch_Topmenu_Label = 101;
public static int Ch_Topmenu_Label_Active = 102;
public static void InitColors () {
NCurses.InitPair (101, CursesColor.WHITE, CursesColor.BLUE);
NCurses.InitPair (102, CursesColor.BLACK, CursesColor.BLUE);
}
}
public class TopMenu {
private nint parentScreen;
private nint childWindow;
public List<MenuItem> MenuItems { get; set; } = new List<MenuItem> ();
public TopMenu (nint screen) {
parentScreen = screen;
NCurses.GetMaxYX (screen, out _, out int width);
childWindow = NCurses.NewWindow (1, width, 0, 0);
}
public void Render (int? activeItem = null) {
int index = 0;
uint
normalColorPair = 0,
activeColorPair = 0;
var hasColor = NCurses.HasColors ();
if (hasColor) {
normalColorPair = NCurses.ColorPair (ColorPairs.Ch_Topmenu_Label);
activeColorPair = NCurses.ColorPair (ColorPairs.Ch_Topmenu_Label_Active);
NCurses.WindowBackground (childWindow, normalColorPair);
NCurses.WindowRefresh (childWindow);
}
NCurses.MoveWindowAddString (childWindow, 0, 0, " ");
foreach (var item in MenuItems) {
var itemIsActive = activeItem is not null && index == (int) activeItem;
if (itemIsActive && hasColor) {
NCurses.WindowAttributeOn (childWindow, activeColorPair);
}
if (itemIsActive) {
NCurses.WindowAddString (childWindow, $"[{item.Label}] ");
} else {
NCurses.WindowAddString (childWindow, $" {item.Label} ");
}
if (itemIsActive && hasColor) {
NCurses.WindowAttributeOff (childWindow, activeColorPair);
}
index ++;
}
NCurses.WindowRefresh (childWindow);
}
public class MenuItem {
public required string Label { get; set; }
public string? HotKey { get; set; }
public MenuType Type { get; set; } = MenuType.Simple;
public List<MenuItem> SubMenuItems = new List<MenuItem> ();
public enum MenuType {
Simple,
TopMenu,
SubMenu,
Spacer
}
}
}

@ -9,7 +9,11 @@ public class CursesWrapper {
public static nint InitNCurses () { public static nint InitNCurses () {
var screen = NCurses.InitScreen (); var screen = NCurses.InitScreen ();
NCurses.SetCursor (0); NCurses.SetCursor (0);
NCurses.UseDefaultColors ();
NCurses.NoEcho (); NCurses.NoEcho ();
NCurses.Raw ();
NCurses.Keypad (screen, true);
NCurses.NoDelay (screen, false);
if (NCurses.HasColors ()) { if (NCurses.HasColors ()) {
NCurses.StartColor (); NCurses.StartColor ();
@ -20,7 +24,7 @@ public class CursesWrapper {
} }
/// <summary> /// <summary>
/// Get the total width of a window /// Get the total width of a window/screen
/// </summary> /// </summary>
/// <param name="window">Window to query</param> /// <param name="window">Window to query</param>
/// <returns>Width of window in columns</returns> /// <returns>Width of window in columns</returns>
@ -38,4 +42,15 @@ public class CursesWrapper {
NCurses.GetMaxYX (window, out int height, out int _); NCurses.GetMaxYX (window, out int height, out int _);
return height; return height;
} }
/// <summary>
/// Compares a key code to a character to test for a held
/// Control key during the key press event
/// </summary>
/// <param name="keyCode"></param>
/// <param name="testChar"></param>
/// <returns></returns>
public static bool KeyPressIsCtrl (int keyCode, int testChar) {
return keyCode == (testChar & 0x1f);
}
} }

@ -0,0 +1,12 @@
namespace SCI.CursesWrapper;
public class InnerWindow : Window {
public InnerWindow (nint rootScreen, InputHandler inputHandler) :
base (
0, 1,
CursesWrapper.GetWidth (rootScreen),
CursesWrapper.GetHeight (rootScreen) - 2,
inputHandler
)
{ }
}

@ -1,3 +1,4 @@
using System.Diagnostics;
using Mindmagma.Curses; using Mindmagma.Curses;
namespace SCI.CursesWrapper; namespace SCI.CursesWrapper;
@ -33,11 +34,6 @@ public class InputHandler {
} }
} }
/// <summary>
/// Root screen to fall back to when no window is active
/// </summary>
private nint rootScreen;
/// <summary> /// <summary>
/// Task running the endless input handler routine /// Task running the endless input handler routine
/// </summary> /// </summary>
@ -48,14 +44,6 @@ public class InputHandler {
/// </summary> /// </summary>
private CancellationTokenSource? listeningRoutineCts; private CancellationTokenSource? listeningRoutineCts;
/// <summary>
/// Class constructor
/// <paramref name="screen">Parent screen this InputHandler is attached to</paramref>
/// </summary>
public InputHandler (nint screen) {
rootScreen = screen;
}
/// <summary> /// <summary>
/// Start listening to key presses and call registered callbacks /// Start listening to key presses and call registered callbacks
/// </summary> /// </summary>
@ -96,59 +84,59 @@ public class InputHandler {
); );
} }
/// <summary>
/// Prepares the window for proper key press reading
/// </summary>
private void WindowSetup () {
if (ActiveWindow is not null) {
NCurses.Keypad (ActiveWindow.WindowId, true);
NCurses.NoDelay (ActiveWindow.WindowId, false);
} else {
NCurses.Keypad (rootScreen, true);
NCurses.NoDelay (rootScreen, false);
}
}
/// <summary>
/// Undoes any changes the setup routine configured for the active window
/// </summary>
private void WindowTeardown () {
if (ActiveWindow is not null) {
NCurses.Keypad (ActiveWindow.WindowId, false);
NCurses.NoDelay (ActiveWindow.WindowId, true);
} else {
NCurses.Keypad (rootScreen, false);
NCurses.NoDelay (rootScreen, true);
}
}
/// <summary> /// <summary>
/// Actual keypress handler being called as Task by StartListening () /// Actual keypress handler being called as Task by StartListening ()
/// </summary> /// </summary>
private void _ListeningRoutine () { private void _ListeningRoutine () {
if (listeningRoutineCts is null) return; if (listeningRoutineCts is null) return;
WindowSetup ();
int keyCode = -1;
while (!listeningRoutineCts.Token.IsCancellationRequested) { while (!listeningRoutineCts.Token.IsCancellationRequested) {
if (OnKeyPress is null) { int keyCode = -1;
Console.Title = "No handlers"; int keyCode2 = -1;
} else {
Console.Title = $"Got {OnKeyPress.GetInvocationList ().Length} handlers";
}
if (ActiveWindow is not null) { if (ActiveWindow is not null) {
keyCode = NCurses.WindowGetChar (ActiveWindow.WindowId); keyCode = NCurses.WindowGetChar (ActiveWindow.WindowId);
// If this block looks yanky; it is.
// This code checks if ESC was pressed and waits for another
// key press for another 10 milliseconds. This is to catch
// key combinations with the Alt key, as it is sent as
// [Alt] + [<key>]
if (keyCode == CursesKey.ESC) {
NCurses.WindowTimeOut (ActiveWindow.WindowId, 10);
try {
keyCode2 = NCurses.WindowGetChar (ActiveWindow.WindowId);
} catch (Exception) {
keyCode2 = -1;
}
NCurses.WindowTimeOut (ActiveWindow.WindowId, -1);
}
} else { } else {
keyCode = NCurses.GetChar (); keyCode = NCurses.GetChar ();
// Janky solution, see above
if (keyCode == CursesKey.ESC) {
NCurses.TimeOut (10);
try {
keyCode2 = NCurses.GetChar ();
} catch (Exception) {
keyCode2 = -1;
}
NCurses.TimeOut (-1);
} }
}
// Do nothing until either key code is larger than -1
if (keyCode < 0 && keyCode2 < 0) continue;
// Do nothing until the key code is larger than -1 // Test if [Alt]+[key] combination was pressed.
if (keyCode < 0) continue; // This is due to [Alt]+[key] being sent as [ESC], [key]
bool isAltCombo = keyCode == CursesKey.ESC && keyCode2 != -1;
if (isAltCombo) {
keyCode = keyCode2;
}
var eventArgs = new NCursesKeyPressEventArgs (keyCode, ActiveWindow); var eventArgs = new NCursesKeyPressEventArgs (keyCode, ActiveWindow, isAltCombo);
// Handle any registered privileged key handlers // Handle any registered privileged key handlers
if (OnKeyPressPrivileged is not null) { if (OnKeyPressPrivileged is not null) {
@ -183,8 +171,12 @@ public class NCursesKeyPressEventArgs : EventArgs {
public bool CancelNextEvent { get; set; } = false; public bool CancelNextEvent { get; set; } = false;
public NCursesKeyPressEventArgs (int keyCode, Window? sourceWindow) { private bool _isAltCombination = false;
public bool IsAltCombination { get { return _isAltCombination; } }
public NCursesKeyPressEventArgs (int keyCode, Window? sourceWindow, bool isAltCombination) {
_keyCode = keyCode; _keyCode = keyCode;
_sourceWindow = sourceWindow; _sourceWindow = sourceWindow;
_isAltCombination = isAltCombination;
} }
} }

@ -5,6 +5,8 @@ using Mindmagma.Curses;
namespace SCI.CursesWrapper; namespace SCI.CursesWrapper;
public class Window { public class Window {
public const int InnerPadding = 1;
/// <summary> /// <summary>
/// Gets or sets the window position on the screen /// Gets or sets the window position on the screen
/// </summary> /// </summary>
@ -15,10 +17,13 @@ public class Window {
} }
set { set {
if (ParentWindow is not null) { if (ParentWindow is not null) {
int addedPadding = ParentWindow.BorderEnabled?
InnerPadding + 1 : 0;
NCurses.WindowMove ( NCurses.WindowMove (
WindowId, WindowId,
ParentWindow.Position.Y + value.Y, ParentWindow.Position.Y + value.Y + addedPadding,
ParentWindow.Position.X + value.X ParentWindow.Position.X + value.X + addedPadding
); );
} else { } else {
NCurses.WindowMove ( NCurses.WindowMove (
@ -54,11 +59,23 @@ public class Window {
} }
set { set {
NCurses.WindowBackground (WindowId, value); NCurses.WindowBackground (WindowId, value);
Redraw (); Draw ();
_backgroundColorId = value; _backgroundColorId = value;
} }
} }
private bool _borderEnabled;
public bool BorderEnabled {
get {
return _borderEnabled;
}
set {
SetBorder (value);
Draw ();
_borderEnabled = value;
}
}
/// <summary> /// <summary>
/// Sets the parent window /// Sets the parent window
/// </summary> /// </summary>
@ -80,7 +97,8 @@ public class Window {
/// <summary> /// <summary>
/// Input handler assigned to this window /// Input handler assigned to this window
/// </summary> /// </summary>
private InputHandler? inputHandler; protected InputHandler? _targetInputHandler;
public InputHandler? TargetInputHandler { get { return _targetInputHandler; }}
/// <summary> /// <summary>
/// Event handler called when this window is active and a key is pressed /// Event handler called when this window is active and a key is pressed
@ -96,20 +114,32 @@ public class Window {
/// <param name="height"></param> /// <param name="height"></param>
/// <param name="parentWindow"></param> /// <param name="parentWindow"></param>
public Window (int x, int y, int width, int height, Window? parentWindow = null) { public Window (int x, int y, int width, int height, Window? parentWindow = null) {
if (parentWindow is not null) { _Initialize (new Point (x, y), new Size (width, height), parentWindow);
_windowId = NCurses.DeriveWindow (
parentWindow.WindowId,
height, width,
y, x
);
} else {
_windowId = NCurses.NewWindow (
height, width,
y, x
);
} }
Redraw (); /// <summary>
/// Create new window by specifying geometry through Point and Size objects
/// </summary>
/// <param name="position"></param>
/// <param name="windowSize"></param>
/// <param name="parentWindow"></param>
public Window (Point position, Size windowSize, Window? parentWindow = null) {
_Initialize (position, windowSize, parentWindow);
}
/// <summary>
/// Create new window by specifying X/Y and Width/Height geometry
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="width"></param>
/// <param name="height"></param>
/// <param name="targetInputHandler"></param>
/// <param name="parentWindow"></param>
public Window (int x, int y, int width, int height, InputHandler targetInputHandler, Window? parentWindow = null) {
_Initialize (new Point (x, y), new Size (width, height), parentWindow);
RegisterInputHandler (targetInputHandler);
SetWindowActive ();
} }
/// <summary> /// <summary>
@ -117,14 +147,32 @@ public class Window {
/// </summary> /// </summary>
/// <param name="position"></param> /// <param name="position"></param>
/// <param name="windowSize"></param> /// <param name="windowSize"></param>
/// <param name="targetInputHandler"></param>
/// <param name="parentWindow"></param> /// <param name="parentWindow"></param>
public Window (Point position, Size windowSize, Window? parentWindow = null) { public Window (Point position, Size windowSize, InputHandler targetInputHandler, Window? parentWindow = null) {
_Initialize (position, windowSize, parentWindow);
RegisterInputHandler (targetInputHandler);
SetWindowActive ();
}
/// <summary>
/// Actual initialization function for this class
/// </summary>
/// <param name="position"></param>
/// <param name="windowSize"></param>
/// <param name="parentWindow"></param>
private void _Initialize (Point position, Size windowSize, Window? parentWindow = null) {
if (parentWindow is not null) { if (parentWindow is not null) {
_windowId = NCurses.SubWindow ( int addedPadding = parentWindow.BorderEnabled?
InnerPadding + 1 : 0;
_windowId = NCurses.DeriveWindow (
parentWindow.WindowId, parentWindow.WindowId,
windowSize.Height, windowSize.Width, windowSize.Height, windowSize.Width,
position.Y, position.X position.Y + addedPadding, position.X + addedPadding
); );
parentWindow.AddChildWindow (this);
} else { } else {
_windowId = NCurses.NewWindow ( _windowId = NCurses.NewWindow (
windowSize.Height, windowSize.Width, windowSize.Height, windowSize.Width,
@ -132,7 +180,20 @@ public class Window {
); );
} }
Redraw (); _position = position;
_windowSize = windowSize;
_parentWindow = parentWindow;
NCurses.Keypad (WindowId, true);
Draw ();
}
/// <summary>
/// Destroys the window and its sub-windows when the Window object is destroyed
/// </summary>
~Window () {
Destroy ();
} }
/// <summary> /// <summary>
@ -149,7 +210,6 @@ public class Window {
/// </summary> /// </summary>
/// <param name="child"></param> /// <param name="child"></param>
public void RemoveChildWindow (Window child) { public void RemoveChildWindow (Window child) {
if (!_childWindows.Contains (child)) return;
_childWindows.Remove (child); _childWindows.Remove (child);
} }
@ -162,15 +222,13 @@ public class Window {
} }
/// <summary> /// <summary>
/// Redraws this window /// Draws this window and all sub windows
/// </summary> /// </summary>
public void Redraw () { public void Draw () {
NCurses.Refresh (); // TODO: Necessary? NCurses.Refresh (); // TODO: Necessary?
if (ChildWindows.Count > 0) {
foreach (var window in ChildWindows) { foreach (var window in ChildWindows) {
window.Redraw (); window.Draw ();
}
} }
if (ParentWindow is not null) ParentWindow.TouchWin (); if (ParentWindow is not null) ParentWindow.TouchWin ();
@ -181,47 +239,54 @@ public class Window {
/// Destroys this window and all children windows /// Destroys this window and all children windows
/// </summary> /// </summary>
public void Destroy () { public void Destroy () {
if (ChildWindows.Count > 0) { // Catch double destroy calls
foreach (var window in _childWindows) { if (WindowId == -1) throw new Exception ("Destroy called twice on object");
foreach (var window in _childWindows.ToList ()) {
window.Destroy (); window.Destroy ();
} }
}
if (inputHandler is not null) inputHandler.ActiveWindow = null;
if (ParentWindow is not null) ParentWindow.RemoveChildWindow (this);
UnregisterInputHandler (); UnregisterInputHandler ();
// Prepare for screen clear
NCurses.WindowBackground (WindowId, NCurses.ColorPair (-1));
SetBorder (false); SetBorder (false);
// Clear window and redraw
NCurses.ClearWindow (WindowId); NCurses.ClearWindow (WindowId);
Console.Title = "About to destroy"; Draw ();
NCurses.DeleteWindow (WindowId); NCurses.DeleteWindow (WindowId);
Console.Title = "Destroyed"; _windowId = -1;
//TODO: Program hangs on DeleteWindow // Ensures sure the parent is updated too
if (ParentWindow is not null) {
ParentWindow.RemoveChildWindow (this);
ParentWindow.Draw ();
}
} }
/// <summary> /// <summary>
/// Register an input handler for this window to attach to OnKeyPress events /// Register an input handler for this window to attach to OnKeyPress events
/// </summary> /// </summary>
/// <param name="inputHandler">InputHandler to register</param> /// <param name="targetInputHandler">InputHandler to register</param>
public void RegisterInputHandler (InputHandler targetInputHandler) { public void RegisterInputHandler (InputHandler inputHandler) {
if (inputHandler is not null) throw new Exception ( if (TargetInputHandler is not null) throw new Exception (
"Another input handler is already registered" "Another input handler is already registered"
); );
inputHandler = targetInputHandler; _targetInputHandler = inputHandler;
inputHandler.OnKeyPress += KeyPressHandler; TargetInputHandler!.OnKeyPress += KeyPressHandler;
} }
/// <summary> /// <summary>
/// Detach from all OnKeyPress events and unset input handler /// Detach from all OnKeyPress events and unset input handler
/// </summary> /// </summary>
public void UnregisterInputHandler () { public void UnregisterInputHandler () {
if (inputHandler is null) return; if (TargetInputHandler is null) return;
inputHandler.OnKeyPress -= KeyPressHandler; if (TargetInputHandler.ActiveWindow == this) TargetInputHandler.ActiveWindow = null;
TargetInputHandler.OnKeyPress -= KeyPressHandler;
} }
/// <summary> /// <summary>
@ -233,7 +298,7 @@ public class Window {
if (e.SourceWindow != this) return; if (e.SourceWindow != this) return;
if (OnKeyPress is not null) { if (OnKeyPress is not null) {
OnKeyPress (sender, e); OnKeyPress (this, e);
} }
} }
@ -241,9 +306,10 @@ public class Window {
/// Tells the input handler this window is active /// Tells the input handler this window is active
/// </summary> /// </summary>
public void SetWindowActive () { public void SetWindowActive () {
if (inputHandler is null) return; if (TargetInputHandler is null) return;
inputHandler.ActiveWindow = this; TargetInputHandler.ActiveWindow = this;
Draw ();
} }
/// <summary> /// <summary>
@ -262,4 +328,28 @@ public class Window {
NCurses.Box (WindowId, ' ', ' '); NCurses.Box (WindowId, ' ', ' ');
} }
} }
/// <summary>
/// Calcaulates the usable inner width of this window
/// </summary>
/// <returns>Usable inner width of window in columns</returns>
public int GetUsableWidth () {
if (BorderEnabled) {
return WindowSize.Width - 1 - InnerPadding;
} else {
return WindowSize.Width;
}
}
/// <summary>
/// Calcaulates the usable inner height of this window
/// </summary>
/// <returns>Usable inner height of window in rows</returns>
public int GetUsableHeight () {
if (BorderEnabled) {
return WindowSize.Height - 1 - InnerPadding;
} else {
return WindowSize.Height;
}
}
} }

@ -1,4 +1,5 @@
using SCI.CursesWrapper; using SCI.CursesWrapper;
using SCI.CursesWrapper.UiElements;
using ANSI_Fahrplan.Screens; using ANSI_Fahrplan.Screens;
using Mindmagma.Curses; using Mindmagma.Curses;
@ -6,8 +7,6 @@ namespace ANSI_Fahrplan;
class Program { class Program {
private static void Main (string [] args) { private static void Main (string [] args) {
//Playground.Run (args);
/** /**
* General procedure: * General procedure:
* - Draw welcome screen * - Draw welcome screen
@ -20,96 +19,76 @@ class Program {
**/ **/
var screen = CursesWrapper.InitNCurses (); var screen = CursesWrapper.InitNCurses ();
//var topLevelWindows = new List<Window> ();
// -- Screen-wide input handler -- // // -- Screen-wide input handler -- //
var inputHandler = new InputHandler (screen); var inputHandler = new InputHandler ();
inputHandler.StartListening (); inputHandler.StartListening ();
// Register quit key on main screen // Register global quit key
inputHandler.OnKeyPress += (object sender, NCursesKeyPressEventArgs e) => { inputHandler.OnKeyPressPrivileged += (object sender, NCursesKeyPressEventArgs e) => {
if (e.SourceWindow is not null) return; if (!CursesWrapper.KeyPressIsCtrl (e.KeyCode, 'q')) return;
if (e.KeyCode != 'q') return;
NCurses.EndWin (); NCurses.EndWin ();
Console.WriteLine ("Bye-bye!"); Console.WriteLine ("Bye-bye!");
Environment.Exit (0); Environment.Exit (0);
}; };
// -- Create menu bar -- // // -- Inner Window -- //
//var topMenu = new TopMenu (screen, CreateMenuItems (screen, screenInputHandler)); //
// This window contains all the dynamic content between
// the menu and status bars
var innerWindow = new InnerWindow (screen, inputHandler) {
BorderEnabled = true
};
// -- Show introduction screen -- // // -- Intro screen -- //
// NCurses.GetMaxYX (screen, out int height, out int width); var introScreen = new IntroScreen (innerWindow);
// var innerWindow = NCurses.NewWindow (height - 2, width - 4, 1, 2); // Close intro screen on any keypress
// NCurses.Box (innerWindow, (char) 0, (char) 0); introScreen.OnKeyPress += (object sender, NCursesKeyPressEventArgs e) => {
((Window) sender).Destroy ();
};
var introScreen = new IntroScreen (screen); introScreen.RegisterInputHandler (inputHandler);
introScreen.SetBorder (true); introScreen.SetWindowActive ();
introScreen.Redraw ();
var parentWindow = new Window (3, 3, 10 * 2, 10) { // Wait until intro screen is closed
BackgroundColorId = ColorSchemes.TextInputField () while (introScreen.WindowId > -1) Thread.Sleep (50); // TODO; Unjank this
};
parentWindow.SetBorder (true);
parentWindow.Redraw ();
var childWindow = new Window (1, 1, 3, 3, parentWindow);
childWindow.SetBorder (true);
childWindow.Redraw ();
inputHandler.OnKeyPress += (object sender, NCursesKeyPressEventArgs e) => {
if (e.SourceWindow is not null) {
NCurses.WindowAddChar (e.SourceWindow.WindowId, e.KeyCode);
return;
} else {
NCurses.AddChar (e.KeyCode);
}
if (e.KeyCode != 'd') return;
parentWindow.Destroy ();
};
NCurses.AddString ("Done drawing"); // -- Create menu bar -- //
NCurses.Refresh (); var topMenu = new TopMenu (screen, inputHandler, CreateMenuItems (innerWindow));
// Wait until the input handler routine stops // Wait until the input handler routine stops
while (true) Thread.Sleep (500); while (inputHandler.IsListening ()) Thread.Sleep (50);
NCurses.EndWin (); NCurses.EndWin ();
Console.WriteLine ("Oh wow, the input handler crashed, that's not supposed to happen. Sorry!"); Console.WriteLine ("Oh wow, the input handler crashed, that's not supposed to happen. Sorry!");
Environment.Exit (1); Environment.Exit (1);
} }
// private static List<MenuItem> CreateMenuItems (nint screen, InputHandler inputHandler) { private static List<MenuItem> CreateMenuItems (InnerWindow innerWindow) {
// var helpItem = new MenuItem ("Help", "F1", inputHandler); var helpItem = new MenuItem ("Help", "F1");
// var upcomingItem = new MenuItem ("Upcoming", "F2", inputHandler); var upcomingItem = new MenuItem ("Upcoming", "b"); //F2
// var byDayItem = new MenuItem ("By Day", "F3", inputHandler); var byDayItem = new MenuItem ("By Day", "n"); // F3
// var byRoomItem = new MenuItem ("By Room", "F4", inputHandler); var byRoomItem = new MenuItem ("By Room", "F4");
// var bySpeakerItem = new MenuItem ("By Speaker", "F5", inputHandler); var bySpeakerItem = new MenuItem ("By Speaker", "F5");
// var quitItem = new MenuItem ("Quit", "q", inputHandler); var byTestItem = new MenuItem ("By Test", "F7");
var quitItem = new MenuItem ("Quit (C-q)");
// quitItem.OnItemActivated += (object? sender, EventArgs e) => {
// NCurses.EndWin (); helpItem.OnItemActivated += (object sender, MenuItemActivatedEventArgs e) => {
// Console.WriteLine ("Bye-bye!");
// Environment.Exit (0); };
// };
return new List<MenuItem> {
// helpItem.OnItemActivated += (object? sender, EventArgs e) => { helpItem,
// InputBox.InputCompleted callback = (string ee) => { upcomingItem,
// NCurses.MoveAddString (3, 3, $"<{ee}>"); byDayItem,
// }; byRoomItem,
// new InputBox().RequestInput (screen, inputHandler, callback, "Please enter some text now:"); bySpeakerItem,
// }; quitItem
};
// return new List<MenuItem> { }
// helpItem,
// upcomingItem,
// byDayItem,
// byRoomItem,
// bySpeakerItem,
// quitItem
// };
// }
} }

@ -1,12 +0,0 @@
namespace ANSI_Fahrplan.Screens;
/// <summary>
/// NCurses Header Window
/// </summary>
public class Header {
public Header () {
}
}

@ -1,20 +1,18 @@
using Mindmagma.Curses;
using SCI.CursesWrapper; using SCI.CursesWrapper;
namespace ANSI_Fahrplan.Screens; namespace ANSI_Fahrplan.Screens;
public class IntroScreen : Window { public class IntroScreen : Window {
private nint infoTextBox;
/// <summary> /// <summary>
/// Pass class constructor through to inherited constructor /// Pass class constructor through to inherited constructor
/// </summary> /// </summary>
/// <param name="parentWindow">Parent window of this window object</param> /// <param name="parentWindow">Parent window of this window object</param>
public IntroScreen (nint screen) : public IntroScreen (Window parentWindow) :
base ( base (
1, 1, 0, 0,
CursesWrapper.GetWidth (screen) - 2, parentWindow.GetUsableWidth (),
CursesWrapper.GetHeight (screen) - 2 parentWindow.GetUsableHeight (),
parentWindow
) )
{ {
string asciiArtPlusText = string asciiArtPlusText =

@ -1,7 +1,6 @@
using SCI.CursesWrapper;
using Mindmagma.Curses; using Mindmagma.Curses;
namespace ANSI_Fahrplan.UiElements; namespace SCI.CursesWrapper.UiElements;
public class InputBox { public class InputBox {
public delegate void InputCompleted (string input); public delegate void InputCompleted (string input);

@ -1,7 +1,6 @@
using SCI.CursesWrapper;
using Mindmagma.Curses; using Mindmagma.Curses;
namespace ANSI_Fahrplan.UiElements; namespace SCI.CursesWrapper.UiElements;
public class MenuItem { public class MenuItem {
private List<MenuItem> _childItems; private List<MenuItem> _childItems;
@ -10,43 +9,75 @@ public class MenuItem {
private string _label; private string _label;
public string Label { get { return _label; }} public string Label { get { return _label; }}
private string _key; private string? _key;
public string Key { get { return _key; }} public string? Key { get { return _key; }}
public delegate void ItemActivated (object sender, EventArgs e); public delegate void ItemActivated (object sender, MenuItemActivatedEventArgs e);
// Event fired when menu item is activated // Event fired when menu item is activated
public event EventHandler? OnItemActivated; public event ItemActivated? OnItemActivated;
private InputHandler _inputHandler; private TopMenu? _parentTopMenuWindow;
public InputHandler InputHandler { get { return _inputHandler; }} public TopMenu? ParentTopMenuWindow { get { return _parentTopMenuWindow; }}
public MenuItem (string label, string key, InputHandler inputHandler, List<MenuItem>? childItems = null) { /// <summary>
/// Class constructor
/// </summary>
/// <param name="label"></param>
/// <param name="key"></param>
/// <param name="childItems"></param>
public MenuItem (string label, string? key = null, List<MenuItem>? childItems = null) {
_childItems = childItems ?? new List<MenuItem> (); _childItems = childItems ?? new List<MenuItem> ();
_key = key; _key = key;
_inputHandler = inputHandler;
_label = label; _label = label;
} }
public void KeyHandler (object sender, NCursesKeyPressEventArgs e) { /// <summary>
/// Sets this menu items parent TopMenu window
/// </summary>
/// <param name="parent">TopMenu window this menu item belongs to</param>
public void AssignParentTopMenuWindow (TopMenu parent) {
if (_parentTopMenuWindow is not null) return;
_parentTopMenuWindow = parent;
}
/// <summary>
/// Called when key on parent TopMenu window is pressed
/// </summary>
/// <param name="sender">Event sender, should be a Window object</param>
/// <param name="e">Key Press Event arguments</param>
public void KeypressHandler (object sender, NCursesKeyPressEventArgs e) {
// Do not listen to key presses if not hotkey is assigned
if (Key is null) return;
// Do not react to key presses unless this menu items belongs to a window
// and has event listeners upon menu item activation
if (OnItemActivated is null) return; if (OnItemActivated is null) return;
if (ParentTopMenuWindow is null) return;
var eventArgs = new MenuItemActivatedEventArgs (ParentTopMenuWindow, this);
if (Key.Length > 1 && Key.StartsWith ("F")) { // Handle F keys if (Key.Length > 1 && Key.StartsWith ("F")) { // Handle F keys
if (e.KeyCode == CursesKey.KEY_F (int.Parse (Key.Substring (1)))) { if (e.KeyCode == CursesKey.KEY_F (int.Parse (Key.Substring (1)))) {
OnItemActivated (this, EventArgs.Empty); OnItemActivated (this, eventArgs);
} }
} else if (Key.Length == 1) { // Handle letters and numbers } else if (Key.Length == 1) { // Handle letters and numbers
if (e.KeyCode == Key [0]) { if (e.KeyCode == Key [0]) {
OnItemActivated (this, EventArgs.Empty); OnItemActivated (this, eventArgs);
} }
} else throw new NotImplementedException ("Currently only F-keys and letters work for Top Menu actions"); } else throw new NotImplementedException ("Currently only F-keys and letters work for Top Menu actions");
} }
}
public void RegisterKeyHandler () { public class MenuItemActivatedEventArgs : EventArgs {
InputHandler.OnKeyPress += KeyHandler; private TopMenu _topMenuWindow;
} public TopMenu TopMenuWindow { get { return _topMenuWindow; }}
private MenuItem _relatedMenuItem;
public MenuItem RelatedMenuItem { get { return _relatedMenuItem; }}
public void UnregisterKeyHandler () { public MenuItemActivatedEventArgs (TopMenu sourceTopMenuWindow, MenuItem relatedMenuItem) {
InputHandler.OnKeyPress -= KeyHandler; _topMenuWindow = sourceTopMenuWindow;
_relatedMenuItem = relatedMenuItem;
} }
} }

@ -1,42 +1,49 @@
using SCI.CursesWrapper;
using Mindmagma.Curses; using Mindmagma.Curses;
using System.Collections.ObjectModel;
namespace ANSI_Fahrplan.UiElements; namespace SCI.CursesWrapper.UiElements;
public class TopMenu : UiElement { public class TopMenu : Window {
private List<MenuItem> _menuItems = new List<MenuItem> (); private List<MenuItem> _menuItems = new List<MenuItem> ();
public List<MenuItem> MenuItems { public ReadOnlyCollection<MenuItem> MenuItems {
get { get {
return _menuItems; return _menuItems.AsReadOnly ();
}
set {
UnregisterItemEventHandlers ();
_menuItems = value;
RegisterItemEventHandlers ();
DrawMenu ();
} }
} }
private MenuItem? activeMenuItem; private MenuItem? activeMenuItem;
public TopMenu (nint screen, List<MenuItem> menuItems) : base (screen) { /// <summary>
NCurses.GetMaxYX (screen, out _, out int width); /// Class constructor
innerWindow = NCurses.NewWindow (1, width, 0, 0); /// </summary>
/// <param name="screen"></param>
/// <param name="targetInputHandler"></param>
/// <param name="menuItems"></param>
public TopMenu (nint screen, InputHandler targetInputHandler, List<MenuItem> menuItems) :
base (0, 0, CursesWrapper.GetWidth (screen), 1, targetInputHandler)
{
var colorSchemeNormal = ColorSchemes.TopMenuNormal (); var colorSchemeNormal = ColorSchemes.TopMenuNormal ();
NCurses.WindowBackground (innerWindow, colorSchemeNormal); BackgroundColorId = colorSchemeNormal;
NCurses.WindowAttributeOn (innerWindow, colorSchemeNormal); NCurses.WindowAttributeOn (WindowId, colorSchemeNormal);
MenuItems = menuItems; _menuItems = menuItems;
foreach (var item in menuItems) {
item.AssignParentTopMenuWindow (this);
}
RegisterItemEventHandlers ();
DrawMenu ();
} }
/// <summary> /// <summary>
/// Register key event handlers for each menu item /// Register key event handlers for each menu item
/// </summary> /// </summary>
private void RegisterItemEventHandlers () { private void RegisterItemEventHandlers () {
if (TargetInputHandler is null) return;
foreach (var item in MenuItems) { foreach (var item in MenuItems) {
item.OnItemActivated += ItemActivatedHandler; item.OnItemActivated += ItemActivatedHandler;
item.RegisterKeyHandler (); TargetInputHandler.OnKeyPressPrivileged += item.KeypressHandler;
} }
} }
@ -44,9 +51,11 @@ public class TopMenu : UiElement {
/// Unregister key event handlers for each menu item /// Unregister key event handlers for each menu item
/// </summary> /// </summary>
private void UnregisterItemEventHandlers () { private void UnregisterItemEventHandlers () {
if (TargetInputHandler is null) return;
foreach (var item in MenuItems) { foreach (var item in MenuItems) {
item.OnItemActivated -= ItemActivatedHandler; item.OnItemActivated -= ItemActivatedHandler;
item.UnregisterKeyHandler (); OnKeyPress -= item.KeypressHandler;
} }
} }
@ -66,7 +75,7 @@ public class TopMenu : UiElement {
/// Draw the top menu with currently configured menu items /// Draw the top menu with currently configured menu items
/// </summary> /// </summary>
private void DrawMenu () { private void DrawMenu () {
NCurses.ClearWindow (innerWindow); NCurses.ClearWindow (WindowId);
var normalColor = ColorSchemes.TopMenuNormal (); var normalColor = ColorSchemes.TopMenuNormal ();
var activeColor = ColorSchemes.TopMenuActive (); var activeColor = ColorSchemes.TopMenuActive ();
@ -77,8 +86,8 @@ public class TopMenu : UiElement {
if (itemIsActive) { if (itemIsActive) {
itemString = "["; itemString = "[";
NCurses.WindowAttributeOff (innerWindow, normalColor); NCurses.WindowAttributeOff (WindowId, normalColor);
NCurses.WindowAttributeOn (innerWindow, activeColor); NCurses.WindowAttributeOn (WindowId, activeColor);
} }
itemString += $"{item.Label}"; itemString += $"{item.Label}";
@ -88,21 +97,16 @@ public class TopMenu : UiElement {
} }
if (itemIsActive) { if (itemIsActive) {
NCurses.WindowAttributeOff (innerWindow, activeColor); NCurses.WindowAttributeOff (WindowId, activeColor);
NCurses.WindowAttributeOn (innerWindow, normalColor); NCurses.WindowAttributeOn (WindowId, normalColor);
itemString += "] "; itemString += "] ";
} else { } else {
itemString += " "; itemString += " ";
} }
NCurses.WindowAddString (innerWindow, itemString); NCurses.WindowAddString (WindowId, itemString);
} }
NCurses.Refresh (); Draw ();
NCurses.WindowRefresh (innerWindow);
}
public void Refresh () {
DrawMenu ();
} }
} }
Loading…
Cancel
Save