Files
Terminal.Gui/Terminal.Gui/Core/ShortcutHelper.cs
BDisp 52f48b2044 Fixes #1384. Added a VisibleChanged event on the View class. (#1385)
* Fixes #1384. Added a VisibleChanged event on the View class.

* Getting the last char.

* Fixes #871. Added Enable property to Responder and added Disabled color for all ColorSchemes.

* Added GetNormalColor method to the View being more readable.

* Fixes the contentBottomRightCorner Enable and Visible.

* Fixes #643. Added AddItemAt and RemoveItem to StatusBar and fixed more bugs.

* Typo fixes.

* Fixes #1387. Allowing the UnitTests project to test internal keywords.

* Fixes #1389. Added a unidirectional feature to the Marquee styles to the ProgressBar.

* Fixes #1394. Added ReflectedType to check for overridden.

* Fixes #1396. Using the Loaded event instead the Ready event.

* Fixes #1402. Only WindowsDriver supports horizontal scroll. (#1403)

* Fixes #1402. Only WindowsDriver supports horizontal scroll.

* Fixes ProcessContinuousButtonPressedAsync on all drivers.

* Fixed internal unit test.

* Fixing warning.

* Fixing Editor scenario error.

* Fixes double and triple click on a touchpad.

* Ensuring reset the counting.

* Allowing touchpad double and triple click with one finger on CursesDriver.

* Allowing touchpad double and triple click with one finger on WindowsDriver.

* Fixes #1414. Fixed multi toplevels and mdi container issues.

* Improving EnsureVisibleBounds and PositionToplevel.

* Added mouseGrabView to the ResetState method.

* Changing namespace.

* Allowing file type on the SaveDialog.

* Fixes SaveDialogs writing the extension twice.
2021-08-11 02:56:30 -07:00

272 lines
7.2 KiB
C#

using NStack;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace Terminal.Gui {
/// <summary>
/// Represents a helper to manipulate shortcut keys used on views.
/// </summary>
public class ShortcutHelper {
private Key shortcut;
/// <summary>
/// This is the global setting that can be used as a global shortcut to invoke the action on the view.
/// </summary>
public virtual Key Shortcut {
get => shortcut;
set {
if (shortcut != value && (PostShortcutValidation (value) || value == Key.Null)) {
shortcut = value;
}
}
}
/// <summary>
/// The keystroke combination used in the <see cref="Shortcut"/> as string.
/// </summary>
public virtual ustring ShortcutTag => GetShortcutTag (shortcut);
/// <summary>
/// The action to run if the <see cref="Shortcut"/> is defined.
/// </summary>
public virtual Action ShortcutAction { get; set; }
/// <summary>
/// Gets the key with all the keys modifiers, especially the shift key that sometimes have to be injected later.
/// </summary>
/// <param name="kb">The <see cref="KeyEvent"/> to check.</param>
/// <returns>The <see cref="KeyEvent.Key"/> with all the keys modifiers.</returns>
public static Key GetModifiersKey (KeyEvent kb)
{
var key = kb.Key;
if (kb.IsAlt && (key & Key.AltMask) == 0) {
key |= Key.AltMask;
}
if (kb.IsCtrl && (key & Key.CtrlMask) == 0) {
key |= Key.CtrlMask;
}
if (kb.IsShift && (key & Key.ShiftMask) == 0) {
key |= Key.ShiftMask;
}
return key;
}
/// <summary>
/// Get the <see cref="Shortcut"/> key as string.
/// </summary>
/// <param name="shortcut">The shortcut key.</param>
/// <param name="delimiter">The delimiter string.</param>
/// <returns></returns>
public static ustring GetShortcutTag (Key shortcut, ustring delimiter = null)
{
if (shortcut == Key.Null) {
return "";
}
var k = shortcut;
if (delimiter == null) {
delimiter = MenuBar.ShortcutDelimiter;
}
ustring tag = ustring.Empty;
var sCut = GetKeyToString (k, out Key knm).ToString ();
if (knm == Key.Unknown) {
k &= ~Key.Unknown;
sCut = GetKeyToString (k, out _).ToString ();
}
if ((k & Key.CtrlMask) != 0) {
tag = "Ctrl";
}
if ((k & Key.ShiftMask) != 0) {
if (!tag.IsEmpty) {
tag += delimiter;
}
tag += "Shift";
}
if ((k & Key.AltMask) != 0) {
if (!tag.IsEmpty) {
tag += delimiter;
}
tag += "Alt";
}
ustring [] keys = ustring.Make (sCut).Split (",");
for (int i = 0; i < keys.Length; i++) {
var key = keys [i].TrimSpace ();
if (key == Key.AltMask.ToString () || key == Key.ShiftMask.ToString () || key == Key.CtrlMask.ToString ()) {
continue;
}
if (!tag.IsEmpty) {
tag += delimiter;
}
if (!key.Contains ("F") && key.Length > 2 && keys.Length == 1) {
k = (uint)Key.AltMask + k;
tag += ((char)k).ToString ();
} else if (key.Length == 2 && key.StartsWith ("D")) {
tag += ((char)key.ElementAt (1)).ToString ();
} else {
tag += key;
}
}
return tag;
}
/// <summary>
/// Return key as string.
/// </summary>
/// <param name="key">The key to extract.</param>
/// <param name="knm">Correspond to the non modifier key.</param>
public static ustring GetKeyToString (Key key, out Key knm)
{
if (key == Key.Null) {
knm = Key.Null;
return "";
}
knm = key;
var mK = key & (Key.AltMask | Key.CtrlMask | Key.ShiftMask);
knm &= ~mK;
for (uint i = (uint)Key.F1; i < (uint)Key.F12; i++) {
if (knm == (Key)i) {
mK |= (Key)i;
}
}
knm &= ~mK;
uint.TryParse (knm.ToString (), out uint c);
var s = mK == Key.Null ? "" : mK.ToString ();
if (s != "" && (knm != Key.Null || c > 0)) {
s += ",";
}
s += c == 0 ? knm == Key.Null ? "" : knm.ToString () : ((char)c).ToString ();
return s;
}
/// <summary>
/// Allows to retrieve a <see cref="Key"/> from a <see cref="ShortcutTag"/>
/// </summary>
/// <param name="tag">The key as string.</param>
/// <param name="delimiter">The delimiter string.</param>
public static Key GetShortcutFromTag (ustring tag, ustring delimiter = null)
{
var sCut = tag;
if (sCut.IsEmpty) {
return default;
}
Key key = Key.Null;
//var hasCtrl = false;
if (delimiter == null) {
delimiter = MenuBar.ShortcutDelimiter;
}
ustring [] keys = sCut.Split (delimiter);
for (int i = 0; i < keys.Length; i++) {
var k = keys [i];
if (k == "Ctrl") {
//hasCtrl = true;
key |= Key.CtrlMask;
} else if (k == "Shift") {
key |= Key.ShiftMask;
} else if (k == "Alt") {
key |= Key.AltMask;
} else if (k.StartsWith ("F") && k.Length > 1) {
int.TryParse (k.Substring (1).ToString (), out int n);
for (uint j = (uint)Key.F1; j <= (uint)Key.F12; j++) {
int.TryParse (((Key)j).ToString ().Substring (1), out int f);
if (f == n) {
key |= (Key)j;
}
}
} else {
key |= (Key)Enum.Parse (typeof (Key), k.ToString ());
}
}
return key;
}
/// <summary>
/// Lookup for a <see cref="Key"/> on range of keys.
/// </summary>
/// <param name="key">The source key.</param>
/// <param name="first">First key in range.</param>
/// <param name="last">Last key in range.</param>
public static bool CheckKeysFlagRange (Key key, Key first, Key last)
{
for (uint i = (uint)first; i < (uint)last; i++) {
if ((key | (Key)i) == key) {
return true;
}
}
return false;
}
/// <summary>
/// Used at key down or key press validation.
/// </summary>
/// <param name="key">The key to validate.</param>
/// <returns><c>true</c> if is valid.<c>false</c>otherwise.</returns>
public static bool PreShortcutValidation (Key key)
{
if ((key & (Key.CtrlMask | Key.ShiftMask | Key.AltMask)) == 0 && !CheckKeysFlagRange (key, Key.F1, Key.F12)) {
return false;
}
return true;
}
/// <summary>
/// Used at key up validation.
/// </summary>
/// <param name="key">The key to validate.</param>
/// <returns><c>true</c> if is valid.<c>false</c>otherwise.</returns>
public static bool PostShortcutValidation (Key key)
{
GetKeyToString (key, out Key knm);
if (CheckKeysFlagRange (key, Key.F1, Key.F12) ||
((key & (Key.CtrlMask | Key.ShiftMask | Key.AltMask)) != 0 && knm != Key.Null && knm != Key.Unknown)) {
return true;
}
return false;
}
/// <summary>
/// Allows a view to run a <see cref="View.ShortcutAction"/> if defined.
/// </summary>
/// <param name="kb">The <see cref="KeyEvent"/></param>
/// <param name="view">The <see cref="View"/></param>
/// <returns><c>true</c> if defined <c>false</c>otherwise.</returns>
public static bool FindAndOpenByShortcut (KeyEvent kb, View view = null)
{
if (view == null) {
return false; }
var key = kb.KeyValue;
var keys = GetModifiersKey (kb);
key |= (int)keys;
foreach (var v in view.Subviews) {
if (v.Shortcut != Key.Null && v.Shortcut == (Key)key) {
var action = v.ShortcutAction;
if (action != null) {
Application.MainLoop.AddIdle (() => {
action ();
return false;
});
}
return true;
}
if (FindAndOpenByShortcut (kb, v)) {
return true;
}
}
return false;
}
}
}