ConsoleKeyMappingTests.cs 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. using static Terminal.Gui.ConsoleDrivers.ConsoleKeyMapping;
  2. namespace Terminal.Gui.ConsoleDrivers;
  3. public class ConsoleKeyMappingTests
  4. {
  5. #if ENABLE_VK_PACKET_NON_WINDOWS
  6. // This test (and the GetConsoleKeyInfoFromKeyCode API) are bogus. They make no sense outside of
  7. // the context of Windows and knowing they keyboard layout. They should be removed.
  8. [Theory]
  9. [InlineData (KeyCode.A | KeyCode.ShiftMask, ConsoleKey.A, KeyCode.A, 'A')]
  10. [InlineData ((KeyCode)'a', ConsoleKey.A, (KeyCode)'a', 'a')]
  11. [InlineData ((KeyCode)'À' | KeyCode.ShiftMask, ConsoleKey.A, (KeyCode)'À', 'À')]
  12. [InlineData ((KeyCode)'à', ConsoleKey.A, (KeyCode)'à', 'à')]
  13. [InlineData ((KeyCode)'Ü' | KeyCode.ShiftMask, ConsoleKey.U, (KeyCode)'Ü', 'Ü')]
  14. [InlineData ((KeyCode)'ü', ConsoleKey.U, (KeyCode)'ü', 'ü')]
  15. [InlineData ((KeyCode)'Ý' | KeyCode.ShiftMask, ConsoleKey.Y, (KeyCode)'Ý', 'Ý')]
  16. [InlineData ((KeyCode)'ý', ConsoleKey.Y, (KeyCode)'ý', 'ý')]
  17. [InlineData ((KeyCode)'!' | KeyCode.ShiftMask, ConsoleKey.D1, (KeyCode)'!', '!')]
  18. [InlineData (KeyCode.D1 | KeyCode.ShiftMask, ConsoleKey.D1, (KeyCode)'!', '!')]
  19. [InlineData (KeyCode.D1, ConsoleKey.D1, KeyCode.D1, '1')]
  20. [InlineData (
  21. (KeyCode)'/' | KeyCode.ShiftMask,
  22. ConsoleKey.D7,
  23. (KeyCode)'/',
  24. '/'
  25. )] // BUGBUG: This is incorrect for ENG keyboards. Shift-7 should be &.
  26. [InlineData (KeyCode.D7 | KeyCode.ShiftMask, ConsoleKey.D7, (KeyCode)'/', '/')]
  27. [InlineData (KeyCode.D7, ConsoleKey.D7, KeyCode.D7, '7')]
  28. [InlineData ((KeyCode)'{' | KeyCode.AltMask | KeyCode.CtrlMask, ConsoleKey.D7, (KeyCode)'{', '{')]
  29. [InlineData ((KeyCode)'?' | KeyCode.ShiftMask, ConsoleKey.Oem4, (KeyCode)'?', '?')]
  30. [InlineData ((KeyCode)'\'', ConsoleKey.Oem4, (KeyCode)'\'', '\'')]
  31. [InlineData (KeyCode.PageDown | KeyCode.ShiftMask, ConsoleKey.PageDown, KeyCode.Null, '\0')]
  32. [InlineData (KeyCode.PageDown, ConsoleKey.PageDown, KeyCode.Null, '\0')]
  33. [InlineData ((KeyCode)'q', ConsoleKey.Q, (KeyCode)'q', 'q')]
  34. [InlineData (KeyCode.F2, ConsoleKey.F2, KeyCode.Null, '\0')]
  35. [InlineData ((KeyCode)'英', ConsoleKey.None, (KeyCode)'英', '英')]
  36. public void GetConsoleKeyInfoFromKeyCode_Tests (
  37. KeyCode keyCode,
  38. ConsoleKey expectedConsoleKey,
  39. KeyCode expectedKeyCode,
  40. char expectedKeyChar
  41. )
  42. {
  43. var consoleKeyInfo = ConsoleKeyMapping.GetConsoleKeyInfoFromKeyCode (keyCode);
  44. Assert.Equal (consoleKeyInfo.Key, expectedConsoleKey);
  45. Assert.Equal ((char)expectedKeyCode, expectedKeyChar);
  46. Assert.Equal (consoleKeyInfo.KeyChar, expectedKeyChar);
  47. }
  48. static object packetLock = new object ();
  49. /// <summary>
  50. /// Sometimes when using remote tools EventKeyRecord sends 'virtual keystrokes'.
  51. /// These are indicated with the wVirtualKeyCode of 231 (VK_PACKET). When we see this code
  52. /// then we need to look to the unicode character (UnicodeChar) instead of the key
  53. /// when telling the rest of the framework what button was pressed. For full details
  54. /// see: https://github.com/gui-cs/Terminal.Gui/issues/2008
  55. /// </summary>
  56. [Theory]
  57. [AutoInitShutdown]
  58. [MemberData (nameof (VKPacket))]
  59. public void TestVKPacket (
  60. uint unicodeCharacter,
  61. bool shift,
  62. bool alt,
  63. bool control,
  64. uint initialVirtualKey,
  65. uint initialScanCode,
  66. KeyCode expectedRemapping,
  67. uint expectedVirtualKey,
  68. uint expectedScanCode
  69. )
  70. {
  71. lock (packetLock)
  72. {
  73. Application._forceFakeConsole = true;
  74. Application.Init ();
  75. ConsoleKeyInfo originalConsoleKeyInfo =
  76. new ConsoleKeyInfo ((char)unicodeCharacter, (ConsoleKey)initialVirtualKey, shift, alt, control);
  77. var encodedChar = ConsoleKeyMapping.EncodeKeyCharForVKPacket (originalConsoleKeyInfo);
  78. ConsoleKeyInfo packetConsoleKeyInfo =
  79. new ConsoleKeyInfo (encodedChar, ConsoleKey.Packet, shift, alt, control);
  80. ConsoleKeyInfo consoleKeyInfo = ConsoleKeyMapping.DecodeVKPacketToKConsoleKeyInfo (packetConsoleKeyInfo);
  81. Assert.Equal (originalConsoleKeyInfo, consoleKeyInfo);
  82. var modifiers = ConsoleKeyMapping.GetModifiers (shift, alt, control);
  83. var scanCode = ConsoleKeyMapping.GetScanCodeFromConsoleKeyInfo (consoleKeyInfo);
  84. Assert.Equal ((uint)consoleKeyInfo.Key, initialVirtualKey);
  85. if (scanCode > 0 && consoleKeyInfo.KeyChar == 0)
  86. {
  87. Assert.Equal (0, (double)consoleKeyInfo.KeyChar);
  88. }
  89. else
  90. {
  91. Assert.Equal (consoleKeyInfo.KeyChar, unicodeCharacter);
  92. }
  93. Assert.Equal ((uint)consoleKeyInfo.Key, expectedVirtualKey);
  94. Assert.Equal (scanCode, initialScanCode);
  95. Assert.Equal (scanCode, expectedScanCode);
  96. var top = Application.Top;
  97. top.KeyDown += (s, e) =>
  98. {
  99. Assert.Equal (Key.ToString (expectedRemapping), Key.ToString (e.KeyCode));
  100. e.Handled = true;
  101. Application.RequestStop ();
  102. };
  103. int iterations = -1;
  104. Application.Iteration += (s, a) =>
  105. {
  106. iterations++;
  107. if (iterations == 0)
  108. {
  109. var keyChar = ConsoleKeyMapping.EncodeKeyCharForVKPacket (consoleKeyInfo);
  110. Application.Driver?.SendKeys (keyChar, ConsoleKey.Packet, shift, alt, control);
  111. }
  112. };
  113. Application.Run ();
  114. Application.Shutdown ();
  115. }
  116. }
  117. public static IEnumerable<object []> VKPacket ()
  118. {
  119. lock (packetLock)
  120. {
  121. // unicodeCharacter, shift, alt, control, initialVirtualKey, initialScanCode, expectedRemapping, expectedVirtualKey, expectedScanCode
  122. yield return new object [] { 'a', false, false, false, 'A', 30, KeyCode.A, 'A', 30 };
  123. yield return new object [] { 'A', true, false, false, 'A', 30, KeyCode.A | KeyCode.ShiftMask, 'A', 30 };
  124. yield return new object [] { 'A', true, true, false, 'A', 30, KeyCode.A | KeyCode.ShiftMask | KeyCode.AltMask, 'A', 30 };
  125. yield return new object [] { 'A', true, true, true, 'A', 30, KeyCode.A | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, 'A', 30 };
  126. yield return new object [] { 'z', false, false, false, 'Z', 44, KeyCode.Z, 'Z', 44 };
  127. yield return new object [] { 'Z', true, false, false, 'Z', 44, KeyCode.Z | KeyCode.ShiftMask, 'Z', 44 };
  128. yield return new object [] { 'Z', true, true, false, 'Z', 44, KeyCode.Z | KeyCode.ShiftMask | KeyCode.AltMask, 'Z', 44 };
  129. yield return new object [] { 'Z', true, true, true, 'Z', 44, KeyCode.Z | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, 'Z', 44 };
  130. yield return new object [] { '英', false, false, false, '\0', 0, (KeyCode)'英', '\0', 0 };
  131. yield return new object [] { '英', true, false, false, '\0', 0, (KeyCode)'英' | KeyCode.ShiftMask, '\0', 0 };
  132. yield return new object [] { '英', true, true, false, '\0', 0, (KeyCode)'英' | KeyCode.ShiftMask | KeyCode.AltMask, '\0', 0 };
  133. yield return new object [] { '英', true, true, true, '\0', 0, (KeyCode)'英' | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '\0', 0 };
  134. yield return new object [] { '+', false, false, false, VK.OEM_PLUS, 26, (KeyCode)'+', VK.OEM_PLUS, 26 };
  135. yield return new object [] { '*', true, false, false, VK.OEM_PLUS, 26, (KeyCode)'*' | KeyCode.ShiftMask, VK.OEM_PLUS, 26 };
  136. yield return new object [] { '+', true, true, false, VK.OEM_PLUS, 26, (KeyCode)'+' | KeyCode.ShiftMask | KeyCode.AltMask, VK.OEM_PLUS, 26 };
  137. yield return new object [] { '+', true, true, true, VK.OEM_PLUS, 26, (KeyCode)'+' | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, VK.OEM_PLUS, 26 };
  138. yield return new object [] { '1', false, false, false, '1', 2, KeyCode.D1, '1', 2 };
  139. yield return new object [] { '!', true, false, false, '1', 2, (KeyCode)'!' | KeyCode.ShiftMask, '1', 2 };
  140. yield return new object [] { '1', true, true, false, '1', 2, KeyCode.D1 | KeyCode.ShiftMask | KeyCode.AltMask, '1', 2 };
  141. yield return new object [] { '1', true, true, true, '1', 2, KeyCode.D1 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '1', 2 };
  142. yield return new object [] { '1', false, true, true, '1', 2, KeyCode.D1 | KeyCode.AltMask | KeyCode.CtrlMask, '1', 2 };
  143. yield return new object [] { '2', false, false, false, '2', 3, KeyCode.D2, '2', 3 };
  144. yield return new object [] { '"', true, false, false, '2', 3, (KeyCode)'"' | KeyCode.ShiftMask, '2', 3 };
  145. yield return new object [] { '2', true, true, false, '2', 3, KeyCode.D2 | KeyCode.ShiftMask | KeyCode.AltMask, '2', 3 };
  146. yield return new object [] { '2', true, true, true, '2', 3, KeyCode.D2 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '2', 3 };
  147. yield return new object [] { '@', false, true, true, '2', 3, (KeyCode)'@' | KeyCode.AltMask | KeyCode.CtrlMask, '2', 3 };
  148. yield return new object [] { '3', false, false, false, '3', 4, KeyCode.D3, '3', 4 };
  149. yield return new object [] { '#', true, false, false, '3', 4, (KeyCode)'#' | KeyCode.ShiftMask, '3', 4 };
  150. yield return new object [] { '3', true, true, false, '3', 4, KeyCode.D3 | KeyCode.ShiftMask | KeyCode.AltMask, '3', 4 };
  151. yield return new object [] { '3', true, true, true, '3', 4, KeyCode.D3 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '3', 4 };
  152. yield return new object [] { '£', false, true, true, '3', 4, (KeyCode)'£' | KeyCode.AltMask | KeyCode.CtrlMask, '3', 4 };
  153. yield return new object [] { '4', false, false, false, '4', 5, KeyCode.D4, '4', 5 };
  154. yield return new object [] { '$', true, false, false, '4', 5, (KeyCode)'$' | KeyCode.ShiftMask, '4', 5 };
  155. yield return new object [] { '4', true, true, false, '4', 5, KeyCode.D4 | KeyCode.ShiftMask | KeyCode.AltMask, '4', 5 };
  156. yield return new object [] { '4', true, true, true, '4', 5, KeyCode.D4 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '4', 5 };
  157. yield return new object [] { '§', false, true, true, '4', 5, (KeyCode)'§' | KeyCode.AltMask | KeyCode.CtrlMask, '4', 5 };
  158. yield return new object [] { '5', false, false, false, '5', 6, KeyCode.D5, '5', 6 };
  159. yield return new object [] { '%', true, false, false, '5', 6, (KeyCode)'%' | KeyCode.ShiftMask, '5', 6 };
  160. yield return new object [] { '5', true, true, false, '5', 6, KeyCode.D5 | KeyCode.ShiftMask | KeyCode.AltMask, '5', 6 };
  161. yield return new object [] { '5', true, true, true, '5', 6, KeyCode.D5 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '5', 6 };
  162. yield return new object [] { '€', false, true, true, '5', 6, (KeyCode)'€' | KeyCode.AltMask | KeyCode.CtrlMask, '5', 6 };
  163. yield return new object [] { '6', false, false, false, '6', 7, KeyCode.D6, '6', 7 };
  164. yield return new object [] { '&', true, false, false, '6', 7, (KeyCode)'&' | KeyCode.ShiftMask, '6', 7 };
  165. yield return new object [] { '6', true, true, false, '6', 7, KeyCode.D6 | KeyCode.ShiftMask | KeyCode.AltMask, '6', 7 };
  166. yield return new object [] { '6', true, true, true, '6', 7, KeyCode.D6 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '6', 7 };
  167. yield return new object [] { '6', false, true, true, '6', 7, KeyCode.D6 | KeyCode.AltMask | KeyCode.CtrlMask, '6', 7 };
  168. yield return new object [] { '7', false, false, false, '7', 8, KeyCode.D7, '7', 8 };
  169. yield return
  170. new object [] { '/', true, false, false, '7', 8, (KeyCode)'/' | KeyCode.ShiftMask, '7', 8 }; // BUGBUG: This is not true for ENG keyboards. Shift-7 is &.
  171. yield return new object [] { '7', true, true, false, '7', 8, KeyCode.D7 | KeyCode.ShiftMask | KeyCode.AltMask, '7', 8 };
  172. yield return new object [] { '7', true, true, true, '7', 8, KeyCode.D7 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '7', 8 };
  173. yield return new object [] { '{', false, true, true, '7', 8, (KeyCode)'{' | KeyCode.AltMask | KeyCode.CtrlMask, '7', 8 };
  174. yield return new object [] { '8', false, false, false, '8', 9, KeyCode.D8, '8', 9 };
  175. yield return new object [] { '(', true, false, false, '8', 9, (KeyCode)'(' | KeyCode.ShiftMask, '8', 9 };
  176. yield return new object [] { '8', true, true, false, '8', 9, KeyCode.D8 | KeyCode.ShiftMask | KeyCode.AltMask, '8', 9 };
  177. yield return new object [] { '8', true, true, true, '8', 9, KeyCode.D8 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '8', 9 };
  178. yield return new object [] { '[', false, true, true, '8', 9, (KeyCode)'[' | KeyCode.AltMask | KeyCode.CtrlMask, '8', 9 };
  179. yield return new object [] { '9', false, false, false, '9', 10, KeyCode.D9, '9', 10 };
  180. yield return new object [] { ')', true, false, false, '9', 10, (KeyCode)')' | KeyCode.ShiftMask, '9', 10 };
  181. yield return new object [] { '9', true, true, false, '9', 10, KeyCode.D9 | KeyCode.ShiftMask | KeyCode.AltMask, '9', 10 };
  182. yield return new object [] { '9', true, true, true, '9', 10, KeyCode.D9 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '9', 10 };
  183. yield return new object [] { ']', false, true, true, '9', 10, (KeyCode)']' | KeyCode.AltMask | KeyCode.CtrlMask, '9', 10 };
  184. yield return new object [] { '0', false, false, false, '0', 11, KeyCode.D0, '0', 11 };
  185. yield return new object [] { '=', true, false, false, '0', 11, (KeyCode)'=' | KeyCode.ShiftMask, '0', 11 };
  186. yield return new object [] { '0', true, true, false, '0', 11, KeyCode.D0 | KeyCode.ShiftMask | KeyCode.AltMask, '0', 11 };
  187. yield return new object [] { '0', true, true, true, '0', 11, KeyCode.D0 | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, '0', 11 };
  188. yield return new object [] { '}', false, true, true, '0', 11, (KeyCode)'}' | KeyCode.AltMask | KeyCode.CtrlMask, '0', 11 };
  189. yield return new object [] { '\'', false, false, false, VK.OEM_4, 12, (KeyCode)'\'', VK.OEM_4, 12 };
  190. yield return new object [] { '?', true, false, false, VK.OEM_4, 12, (KeyCode)'?' | KeyCode.ShiftMask, VK.OEM_4, 12 };
  191. yield return new object [] { '\'', true, true, false, VK.OEM_4, 12, (KeyCode)'\'' | KeyCode.ShiftMask | KeyCode.AltMask, VK.OEM_4, 12 };
  192. yield return new object [] { '\'', true, true, true, VK.OEM_4, 12, (KeyCode)'\'' | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, VK.OEM_4, 12 };
  193. yield return new object [] { '«', false, false, false, VK.OEM_6, 13, (KeyCode)'«', VK.OEM_6, 13 };
  194. yield return new object [] { '»', true, false, false, VK.OEM_6, 13, (KeyCode)'»' | KeyCode.ShiftMask, VK.OEM_6, 13 };
  195. yield return new object [] { '«', true, true, false, VK.OEM_6, 13, (KeyCode)'«' | KeyCode.ShiftMask | KeyCode.AltMask, VK.OEM_6, 13 };
  196. yield return new object [] { '«', true, true, true, VK.OEM_6, 13, (KeyCode)'«' | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, VK.OEM_6, 13 };
  197. yield return new object [] { 'á', false, false, false, 'A', 30, (KeyCode)'á', 'A', 30 };
  198. yield return new object [] { 'Á', true, false, false, 'A', 30, (KeyCode)'Á' | KeyCode.ShiftMask, 'A', 30 };
  199. yield return new object [] { 'à', false, false, false, 'A', 30, (KeyCode)'à', 'A', 30 };
  200. yield return new object [] { 'À', true, false, false, 'A', 30, (KeyCode)'À' | KeyCode.ShiftMask, 'A', 30 };
  201. yield return new object [] { 'é', false, false, false, 'E', 18, (KeyCode)'é', 'E', 18 };
  202. yield return new object [] { 'É', true, false, false, 'E', 18, (KeyCode)'É' | KeyCode.ShiftMask, 'E', 18 };
  203. yield return new object [] { 'è', false, false, false, 'E', 18, (KeyCode)'è', 'E', 18 };
  204. yield return new object [] { 'È', true, false, false, 'E', 18, (KeyCode)'È' | KeyCode.ShiftMask, 'E', 18 };
  205. yield return new object [] { 'í', false, false, false, 'I', 23, (KeyCode)'í', 'I', 23 };
  206. yield return new object [] { 'Í', true, false, false, 'I', 23, (KeyCode)'Í' | KeyCode.ShiftMask, 'I', 23 };
  207. yield return new object [] { 'ì', false, false, false, 'I', 23, (KeyCode)'ì', 'I', 23 };
  208. yield return new object [] { 'Ì', true, false, false, 'I', 23, (KeyCode)'Ì' | KeyCode.ShiftMask, 'I', 23 };
  209. yield return new object [] { 'ó', false, false, false, 'O', 24, (KeyCode)'ó', 'O', 24 };
  210. yield return new object [] { 'Ó', true, false, false, 'O', 24, (KeyCode)'Ó' | KeyCode.ShiftMask, 'O', 24 };
  211. yield return new object [] { 'ò', false, false, false, 'O', 24, (KeyCode)'ò', 'O', 24 };
  212. yield return new object [] { 'Ò', true, false, false, 'O', 24, (KeyCode)'Ò' | KeyCode.ShiftMask, 'O', 24 };
  213. yield return new object [] { 'ú', false, false, false, 'U', 22, (KeyCode)'ú', 'U', 22 };
  214. yield return new object [] { 'Ú', true, false, false, 'U', 22, (KeyCode)'Ú' | KeyCode.ShiftMask, 'U', 22 };
  215. yield return new object [] { 'ù', false, false, false, 'U', 22, (KeyCode)'ù', 'U', 22 };
  216. yield return new object [] { 'Ù', true, false, false, 'U', 22, (KeyCode)'Ù' | KeyCode.ShiftMask, 'U', 22 };
  217. yield return new object [] { 'ö', false, false, false, 'O', 24, (KeyCode)'ö', 'O', 24 };
  218. yield return new object [] { 'Ö', true, false, false, 'O', 24, (KeyCode)'Ö' | KeyCode.ShiftMask, 'O', 24 };
  219. yield return new object [] { '<', false, false, false, VK.OEM_102, 86, (KeyCode)'<', VK.OEM_102, 86 };
  220. yield return new object [] { '>', true, false, false, VK.OEM_102, 86, (KeyCode)'>' | KeyCode.ShiftMask, VK.OEM_102, 86 };
  221. yield return new object [] { '<', true, true, false, VK.OEM_102, 86, (KeyCode)'<' | KeyCode.ShiftMask | KeyCode.AltMask, VK.OEM_102, 86 };
  222. yield return new object [] { '<', true, true, true, VK.OEM_102, 86, (KeyCode)'<' | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, VK.OEM_102, 86 };
  223. yield return new object [] { 'ç', false, false, false, VK.OEM_3, 39, (KeyCode)'ç', VK.OEM_3, 39 };
  224. yield return new object [] { 'Ç', true, false, false, VK.OEM_3, 39, (KeyCode)'Ç' | KeyCode.ShiftMask, VK.OEM_3, 39 };
  225. yield return new object [] { 'ç', true, true, false, VK.OEM_3, 39, (KeyCode)'ç' | KeyCode.ShiftMask | KeyCode.AltMask, VK.OEM_3, 39 };
  226. yield return new object [] { 'ç', true, true, true, VK.OEM_3, 39, (KeyCode)'ç' | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, VK.OEM_3, 39 };
  227. yield return new object [] { '¨', false, true, true, VK.OEM_PLUS, 26, (KeyCode)'¨' | KeyCode.AltMask | KeyCode.CtrlMask, VK.OEM_PLUS, 26 };
  228. yield return new object [] { '\0', false, false, false, VK.PRIOR, 73, KeyCode.PageUp, VK.PRIOR, 73 };
  229. yield return new object [] { '\0', true, false, false, VK.PRIOR, 73, KeyCode.PageUp | KeyCode.ShiftMask, VK.PRIOR, 73 };
  230. yield return new object [] { '\0', true, true, false, VK.PRIOR, 73, KeyCode.PageUp | KeyCode.ShiftMask | KeyCode.AltMask, VK.PRIOR, 73 };
  231. yield return new object [] { '\0', true, true, true, VK.PRIOR, 73, KeyCode.PageUp | KeyCode.ShiftMask | KeyCode.AltMask | KeyCode.CtrlMask, VK.PRIOR, 73 };
  232. yield return new object [] { '~', false, false, false, VK.SPACE, 57, (KeyCode)'~', VK.SPACE, 57 };
  233. yield return new object [] { '^', false, false, false, VK.SPACE, 57, (KeyCode)'^', VK.SPACE, 57 };
  234. }
  235. }
  236. [Theory]
  237. [InlineData ('a', ConsoleKey.A, 'a', ConsoleKey.A)]
  238. [InlineData ('A', ConsoleKey.A, 'A', ConsoleKey.A)]
  239. [InlineData ('á', ConsoleKey.A, 'á', ConsoleKey.A)]
  240. [InlineData ('Á', ConsoleKey.A, 'Á', ConsoleKey.A)]
  241. [InlineData ('à', ConsoleKey.A, 'à', ConsoleKey.A)]
  242. [InlineData ('À', ConsoleKey.A, 'À', ConsoleKey.A)]
  243. [InlineData ('5', ConsoleKey.D5, '5', ConsoleKey.D5)]
  244. [InlineData ('%', ConsoleKey.D5, '%', ConsoleKey.D5)]
  245. [InlineData ('€', ConsoleKey.D5, '€', ConsoleKey.D5)]
  246. [InlineData ('?', ConsoleKey.Oem4, '?', ConsoleKey.Oem4)]
  247. [InlineData ('\'', ConsoleKey.Oem4, '\'', ConsoleKey.Oem4)]
  248. [InlineData ('q', ConsoleKey.Q, 'q', ConsoleKey.Q)]
  249. [InlineData ('\0', ConsoleKey.F2, '\0', ConsoleKey.F2)]
  250. [InlineData ('英', ConsoleKey.None, '英', ConsoleKey.None)]
  251. [InlineData ('´', ConsoleKey.None, '´', ConsoleKey.Oem1)]
  252. [InlineData ('`', ConsoleKey.None, '`', ConsoleKey.Oem1)]
  253. //[InlineData ('~', ConsoleKey.None, '~', ConsoleKey.Oem2)]
  254. //[InlineData ('^', ConsoleKey.None, '^', ConsoleKey.Oem2)] // BUGBUG: '^' is Shift-6 on ENG keyboard,not Oem2.
  255. // For the US standard keyboard, Oem2 is the /? key
  256. public void EncodeKeyCharForVKPacket_DecodeVKPacketToKConsoleKeyInfo (
  257. char keyChar,
  258. ConsoleKey consoleKey,
  259. char expectedChar,
  260. ConsoleKey expectedConsoleKey
  261. )
  262. {
  263. var consoleKeyInfo = new ConsoleKeyInfo (keyChar, consoleKey, false, false, false);
  264. var encodedKeyChar = ConsoleKeyMapping.EncodeKeyCharForVKPacket (consoleKeyInfo);
  265. var encodedConsoleKeyInfo = new ConsoleKeyInfo (encodedKeyChar, ConsoleKey.Packet, false, false, false);
  266. var decodedConsoleKeyInfo = ConsoleKeyMapping.DecodeVKPacketToKConsoleKeyInfo (encodedConsoleKeyInfo);
  267. Assert.Equal (consoleKeyInfo.Key, consoleKey);
  268. Assert.Equal (expectedConsoleKey, decodedConsoleKeyInfo.Key);
  269. Assert.Equal (expectedChar, decodedConsoleKeyInfo.KeyChar);
  270. }
  271. [Theory]
  272. [InlineData ((KeyCode)'a', false, ConsoleKey.A, 'a')]
  273. [InlineData (KeyCode.A | KeyCode.ShiftMask, false, ConsoleKey.A, 'A')]
  274. [InlineData ((KeyCode)'á', false, ConsoleKey.A, 'á')]
  275. [InlineData ((KeyCode)'Á' | KeyCode.ShiftMask, false, ConsoleKey.A, 'Á')]
  276. [InlineData ((KeyCode)'à', false, ConsoleKey.A, 'à')]
  277. [InlineData ((KeyCode)'À' | KeyCode.ShiftMask, false, ConsoleKey.A, 'À')]
  278. [InlineData (KeyCode.D5, false, ConsoleKey.D5, '5')]
  279. [InlineData ((KeyCode)'%' | KeyCode.ShiftMask, false, ConsoleKey.D5, '%')]
  280. //[InlineData ((KeyCode)'€' | KeyCode.AltMask | KeyCode.CtrlMask, false, ConsoleKey.D5, '€')] // Bogus test. This is not true on ENG keyboard layout.
  281. [InlineData ((KeyCode)'?' | KeyCode.ShiftMask, false, ConsoleKey.Oem4, '?')]
  282. [InlineData ((KeyCode)'\'', false, ConsoleKey.Oem4, '\'')]
  283. [InlineData ((KeyCode)'q', false, ConsoleKey.Q, 'q')]
  284. [InlineData (KeyCode.F2, true, ConsoleKey.F2, 'q')]
  285. [InlineData ((KeyCode)'英', false, ConsoleKey.None, '英')]
  286. [InlineData (KeyCode.Enter, true, ConsoleKey.Enter, '\r')]
  287. public void MapKeyCodeToConsoleKey_GetKeyCharFromUnicodeChar (
  288. KeyCode keyCode,
  289. bool expectedIsConsoleKey,
  290. ConsoleKey expectedConsoleKey,
  291. char expectedConsoleKeyChar
  292. )
  293. {
  294. var modifiers = ConsoleKeyMapping.MapToConsoleModifiers (keyCode);
  295. var consoleKey = ConsoleKeyMapping.MapKeyCodeToConsoleKey (keyCode, out bool isConsoleKey);
  296. if (isConsoleKey)
  297. {
  298. Assert.True (isConsoleKey == expectedIsConsoleKey);
  299. Assert.Equal (expectedConsoleKey, (ConsoleKey)consoleKey);
  300. Assert.Equal (expectedConsoleKeyChar, consoleKey);
  301. }
  302. else
  303. {
  304. var keyChar =
  305. ConsoleKeyMapping.GetKeyCharFromUnicodeChar (
  306. consoleKey,
  307. modifiers,
  308. out consoleKey,
  309. out _,
  310. isConsoleKey
  311. );
  312. Assert.True (isConsoleKey == expectedIsConsoleKey);
  313. Assert.Equal (expectedConsoleKey, (ConsoleKey)consoleKey);
  314. Assert.Equal (expectedConsoleKeyChar, keyChar);
  315. }
  316. }
  317. #endif
  318. [Theory]
  319. [InlineData ('a', ConsoleKey.A, false, false, false, (KeyCode)'a')]
  320. [InlineData ('A', ConsoleKey.A, true, false, false, KeyCode.A | KeyCode.ShiftMask)]
  321. [InlineData ('á', ConsoleKey.A, false, false, false, (KeyCode)'á')]
  322. [InlineData ('Á', ConsoleKey.A, true, false, false, (KeyCode)'Á' | KeyCode.ShiftMask)]
  323. [InlineData ('à', ConsoleKey.A, false, false, false, (KeyCode)'à')]
  324. [InlineData ('À', ConsoleKey.A, true, false, false, (KeyCode)'À' | KeyCode.ShiftMask)]
  325. [InlineData ('5', ConsoleKey.D5, false, false, false, KeyCode.D5)]
  326. [InlineData ('%', ConsoleKey.D5, true, false, false, (KeyCode)'%' | KeyCode.ShiftMask)]
  327. [InlineData ('€', ConsoleKey.D5, false, true, true, (KeyCode)'€' | KeyCode.AltMask | KeyCode.CtrlMask)]
  328. [InlineData ('?', ConsoleKey.Oem4, true, false, false, (KeyCode)'?' | KeyCode.ShiftMask)]
  329. [InlineData ('\'', ConsoleKey.Oem4, false, false, false, (KeyCode)'\'')]
  330. [InlineData ('q', ConsoleKey.Q, false, false, false, (KeyCode)'q')]
  331. [InlineData ('\0', ConsoleKey.F2, false, false, false, KeyCode.F2)]
  332. [InlineData ('英', ConsoleKey.None, false, false, false, (KeyCode)'英')]
  333. [InlineData ('\r', ConsoleKey.Enter, false, false, false, KeyCode.Enter)]
  334. public void MapConsoleKeyInfoToKeyCode_Also_Return_Modifiers (
  335. char keyChar,
  336. ConsoleKey consoleKey,
  337. bool shift,
  338. bool alt,
  339. bool control,
  340. KeyCode expectedKeyCode
  341. )
  342. {
  343. var consoleKeyInfo = new ConsoleKeyInfo (keyChar, consoleKey, shift, alt, control);
  344. KeyCode keyCode = MapConsoleKeyInfoToKeyCode (consoleKeyInfo);
  345. Assert.Equal (keyCode, expectedKeyCode);
  346. }
  347. [Theory]
  348. [InlineData ('a', false, false, false, (KeyCode)'a')]
  349. [InlineData ('A', true, false, false, KeyCode.A | KeyCode.ShiftMask)]
  350. [InlineData ('á', false, false, false, (KeyCode)'á')]
  351. [InlineData ('Á', true, false, false, (KeyCode)'Á' | KeyCode.ShiftMask)]
  352. [InlineData ('à', false, false, false, (KeyCode)'à')]
  353. [InlineData ('À', true, false, false, (KeyCode)'À' | KeyCode.ShiftMask)]
  354. [InlineData ('5', false, false, false, KeyCode.D5)]
  355. [InlineData ('%', true, false, false, (KeyCode)'%' | KeyCode.ShiftMask)]
  356. [InlineData ('€', false, true, true, (KeyCode)'€' | KeyCode.AltMask | KeyCode.CtrlMask)]
  357. [InlineData ('?', true, false, false, (KeyCode)'?' | KeyCode.ShiftMask)]
  358. [InlineData ('\'', false, false, false, (KeyCode)'\'')]
  359. [InlineData ('q', false, false, false, (KeyCode)'q')]
  360. [InlineData ((uint)KeyCode.F2, false, false, false, KeyCode.F2)]
  361. [InlineData ('英', false, false, false, (KeyCode)'英')]
  362. [InlineData ('\r', false, false, false, KeyCode.Enter)]
  363. [InlineData ('\n', false, false, false, (KeyCode)'\n')]
  364. public void MapToKeyCodeModifiers_Tests (
  365. uint keyChar,
  366. bool shift,
  367. bool alt,
  368. bool control,
  369. KeyCode expectedKeyCode
  370. )
  371. {
  372. ConsoleModifiers modifiers = GetModifiers (shift, alt, control);
  373. var keyCode = (KeyCode)keyChar;
  374. keyCode = MapToKeyCodeModifiers (modifiers, keyCode);
  375. Assert.Equal (keyCode, expectedKeyCode);
  376. }
  377. [Theory]
  378. [MemberData (nameof (GetScanCodeData))]
  379. public void GetScanCodeFromConsoleKeyInfo_Tests (
  380. char keyChar,
  381. ConsoleKey consoleKey,
  382. bool shift,
  383. bool alt,
  384. bool control,
  385. uint expectedScanCode
  386. )
  387. {
  388. var consoleKeyInfo = new ConsoleKeyInfo (keyChar, consoleKey, shift, alt, control);
  389. uint scanCode = GetScanCodeFromConsoleKeyInfo (consoleKeyInfo);
  390. Assert.Equal (scanCode, expectedScanCode);
  391. }
  392. public static IEnumerable<object []> GetScanCodeData ()
  393. {
  394. yield return ['a', ConsoleKey.A, false, false, false, 30];
  395. yield return ['A', ConsoleKey.A, true, false, false, 30];
  396. yield return ['á', ConsoleKey.A, false, false, false, 30];
  397. yield return ['Á', ConsoleKey.A, true, false, false, 30];
  398. yield return ['à', ConsoleKey.A, false, false, false, 30];
  399. yield return ['À', ConsoleKey.A, true, false, false, 30];
  400. yield return ['0', ConsoleKey.D0, false, false, false, 11];
  401. yield return ['=', ConsoleKey.D0, true, false, false, 11];
  402. yield return ['}', ConsoleKey.D0, false, true, true, 11];
  403. yield return ['1', ConsoleKey.D1, false, false, false, 2];
  404. yield return ['!', ConsoleKey.D1, true, false, false, 2];
  405. yield return ['2', ConsoleKey.D2, false, false, false, 3];
  406. yield return ['"', ConsoleKey.D2, true, false, false, 3];
  407. yield return ['@', ConsoleKey.D2, false, true, true, 3];
  408. yield return ['3', ConsoleKey.D3, false, false, false, 4];
  409. yield return ['#', ConsoleKey.D3, true, false, false, 4];
  410. yield return ['£', ConsoleKey.D3, false, true, true, 4];
  411. yield return ['4', ConsoleKey.D4, false, false, false, 5];
  412. yield return ['$', ConsoleKey.D4, true, false, false, 5];
  413. yield return ['§', ConsoleKey.D4, false, true, true, 5];
  414. yield return ['5', ConsoleKey.D5, false, false, false, 6];
  415. yield return ['%', ConsoleKey.D5, true, false, false, 6];
  416. yield return ['€', ConsoleKey.D5, false, true, true, 6];
  417. yield return ['6', ConsoleKey.D6, false, false, false, 7];
  418. yield return ['&', ConsoleKey.D6, true, false, false, 7];
  419. yield return ['7', ConsoleKey.D7, false, false, false, 8];
  420. yield return ['/', ConsoleKey.D7, true, false, false, 8];
  421. yield return ['{', ConsoleKey.D7, false, true, true, 8];
  422. yield return ['8', ConsoleKey.D8, false, false, false, 9];
  423. yield return ['(', ConsoleKey.D8, true, false, false, 9];
  424. yield return ['[', ConsoleKey.D8, false, true, true, 9];
  425. yield return ['9', ConsoleKey.D9, false, false, false, 10];
  426. yield return [')', ConsoleKey.D9, true, false, false, 10];
  427. yield return [']', ConsoleKey.D9, false, true, true, 10];
  428. yield return ['´', ConsoleKey.Oem1, false, false, false, 27];
  429. yield return ['`', ConsoleKey.Oem1, true, false, false, 27];
  430. yield return ['~', ConsoleKey.Oem2, false, false, false, 43];
  431. yield return ['^', ConsoleKey.Oem2, true, false, false, 43];
  432. yield return ['ç', ConsoleKey.Oem3, false, false, false, 39];
  433. yield return ['Ç', ConsoleKey.Oem3, true, false, false, 39];
  434. yield return ['\'', ConsoleKey.Oem4, false, false, false, 12];
  435. yield return ['?', ConsoleKey.Oem4, true, false, false, 12];
  436. yield return ['\\', ConsoleKey.Oem5, false, true, true, 41];
  437. yield return ['|', ConsoleKey.Oem5, true, false, false, 41];
  438. yield return ['«', ConsoleKey.Oem6, false, true, true, 13];
  439. yield return ['»', ConsoleKey.Oem6, true, false, false, 13];
  440. yield return ['º', ConsoleKey.Oem7, false, true, true, 40];
  441. yield return ['ª', ConsoleKey.Oem7, true, false, false, 40];
  442. yield return ['+', ConsoleKey.OemPlus, false, true, true, 26];
  443. yield return ['*', ConsoleKey.OemPlus, true, false, false, 26];
  444. yield return ['¨', ConsoleKey.OemPlus, false, true, true, 26];
  445. yield return [',', ConsoleKey.OemComma, false, true, true, 51];
  446. yield return [';', ConsoleKey.OemComma, true, false, false, 51];
  447. yield return ['.', ConsoleKey.OemPeriod, false, true, true, 52];
  448. yield return [':', ConsoleKey.OemPeriod, true, false, false, 52];
  449. yield return ['-', ConsoleKey.OemMinus, false, true, true, 53];
  450. yield return ['_', ConsoleKey.OemMinus, true, false, false, 53];
  451. yield return ['q', ConsoleKey.Q, false, false, false, 16];
  452. yield return ['\0', ConsoleKey.F2, false, false, false, 60];
  453. yield return ['英', ConsoleKey.None, false, false, false, 0];
  454. yield return ['英', ConsoleKey.None, true, false, false, 0];
  455. }
  456. [Theory]
  457. [MemberData (nameof (UnShiftedChars))]
  458. public void GetKeyChar_Shifted_Char_From_UnShifted_Char (
  459. char unicodeChar,
  460. char expectedKeyChar,
  461. KeyCode expectedKeyCode
  462. )
  463. {
  464. ConsoleModifiers modifiers = GetModifiers (true, false, false);
  465. uint keyChar = GetKeyChar (unicodeChar, modifiers);
  466. Assert.Equal (keyChar, expectedKeyChar);
  467. var keyCode = (KeyCode)keyChar;
  468. keyCode = MapToKeyCodeModifiers (modifiers, keyCode);
  469. Assert.Equal (keyCode, expectedKeyCode);
  470. }
  471. public static IEnumerable<object []> UnShiftedChars =>
  472. new List<object []>
  473. {
  474. new object[] { 'a', 'A', KeyCode.A | KeyCode.ShiftMask },
  475. new object[] { 'z', 'Z', KeyCode.Z | KeyCode.ShiftMask },
  476. new object[] { 'á', 'Á', (KeyCode)'Á' | KeyCode.ShiftMask },
  477. new object[] { 'à', 'À', (KeyCode)'À' | KeyCode.ShiftMask },
  478. new object[]{ 'ý', 'Ý', (KeyCode)'Ý' | KeyCode.ShiftMask },
  479. new object[]{ '1', '!', (KeyCode)'!' | KeyCode.ShiftMask },
  480. new object[]{ '2', '"', (KeyCode)'"' | KeyCode.ShiftMask },
  481. new object[]{ '3', '#', (KeyCode)'#' | KeyCode.ShiftMask },
  482. new object[]{ '4', '$', (KeyCode)'$' | KeyCode.ShiftMask },
  483. new object[]{ '5', '%', (KeyCode)'%' | KeyCode.ShiftMask },
  484. new object[]{ '6', '&', (KeyCode)'&' | KeyCode.ShiftMask },
  485. new object[]{ '7', '/', (KeyCode)'/' | KeyCode.ShiftMask },
  486. new object[]{ '8', '(', (KeyCode)'(' | KeyCode.ShiftMask },
  487. new object[]{ '9', ')', (KeyCode)')' | KeyCode.ShiftMask },
  488. new object[]{ '0', '=', (KeyCode)'=' | KeyCode.ShiftMask },
  489. new object[]{ '\\', '|', (KeyCode)'|' | KeyCode.ShiftMask },
  490. new object[]{ '\'', '?', (KeyCode)'?' | KeyCode.ShiftMask },
  491. new object[]{ '«', '»', (KeyCode)'»' | KeyCode.ShiftMask },
  492. new object[]{ '+', '*', (KeyCode)'*' | KeyCode.ShiftMask },
  493. new object[]{ '´', '`', (KeyCode)'`' | KeyCode.ShiftMask },
  494. new object[]{ 'º', 'ª', (KeyCode)'ª' | KeyCode.ShiftMask },
  495. new object[]{ '~', '^', (KeyCode)'^' | KeyCode.ShiftMask },
  496. new object[]{ '<', '>', (KeyCode)'>' | KeyCode.ShiftMask },
  497. new object[]{ ',', ';', (KeyCode)';' | KeyCode.ShiftMask },
  498. new object[]{ '.', ':', (KeyCode)':' | KeyCode.ShiftMask },
  499. new object[]{ '-', '_', (KeyCode)'_' | KeyCode.ShiftMask },
  500. };
  501. [Theory]
  502. [MemberData (nameof (ShiftedChars))]
  503. public void GetKeyChar_UnShifted_Char_From_Shifted_Char (
  504. char unicodeChar,
  505. char expectedKeyChar,
  506. KeyCode expectedKeyCode
  507. )
  508. {
  509. ConsoleModifiers modifiers = GetModifiers (false, false, false);
  510. uint keyChar = GetKeyChar (unicodeChar, modifiers);
  511. Assert.Equal (keyChar, expectedKeyChar);
  512. var keyCode = (KeyCode)keyChar;
  513. keyCode = MapToKeyCodeModifiers (modifiers, keyCode);
  514. Assert.Equal (keyCode, expectedKeyCode);
  515. }
  516. public static IEnumerable<object []> ShiftedChars =>
  517. new List<object []>
  518. {
  519. new object[] { 'A', 'a', (KeyCode)'a' },
  520. new object[] { 'Z', 'z', (KeyCode)'z' },
  521. new object[] { 'Á', 'á', (KeyCode)'á' },
  522. new object[] { 'À', 'à', (KeyCode)'à' },
  523. new object[] { 'Ý', 'ý', (KeyCode)'ý' },
  524. new object[] { '!', '1', KeyCode.D1 },
  525. new object[] { '"', '2', KeyCode.D2 },
  526. new object[] { '#', '3', KeyCode.D3 },
  527. new object[] { '$', '4', KeyCode.D4 },
  528. new object[] { '%', '5', KeyCode.D5 },
  529. new object[] { '&', '6', KeyCode.D6 },
  530. new object[] { '/', '7', KeyCode.D7 },
  531. new object[] { '(', '8', KeyCode.D8 },
  532. new object[] { ')', '9', KeyCode.D9 },
  533. new object[] { '=', '0', KeyCode.D0 },
  534. new object[] { '|', '\\', (KeyCode)'\\' },
  535. new object[] { '?', '\'', (KeyCode)'\'' },
  536. new object[] { '»', '«', (KeyCode)'«' },
  537. new object[] { '*', '+', (KeyCode)'+' },
  538. new object[] { '`', '´', (KeyCode)'´' },
  539. new object[] { 'ª', 'º', (KeyCode)'º' },
  540. new object[] { '^', '~', (KeyCode)'~' },
  541. new object[] { '>', '<', (KeyCode)'<' },
  542. new object[] { ';', ',', (KeyCode)',' },
  543. new object[] { ':', '.', (KeyCode)'.' },
  544. new object[] { '_', '-', (KeyCode)'-' }
  545. };
  546. }