mirror of
				https://github.com/Ryujinx/Opentk.git
				synced 2025-10-26 13:37:04 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			366 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			366 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| #region --- License ---
 | ||
| /* Copyright (c) 2007 Stefanos Apostolopoulos
 | ||
|  * See license.txt for license info
 | ||
|  */
 | ||
| #endregion
 | ||
| 
 | ||
| #region --- Using directives ---
 | ||
| 
 | ||
| using System;
 | ||
| using System.Collections.Generic;
 | ||
| using System.Text;
 | ||
| using System.Runtime.InteropServices;
 | ||
| using System.Diagnostics;
 | ||
| using Microsoft.Win32;
 | ||
| using OpenTK.Input;
 | ||
| 
 | ||
| #endregion
 | ||
| 
 | ||
| namespace OpenTK.Platform.Windows
 | ||
| {
 | ||
|     internal class WinRawKeyboard : IKeyboardDriver, IDisposable
 | ||
|     {
 | ||
|         private List<Keyboard> keyboards = new List<Keyboard>();
 | ||
|         private IntPtr windowHandle;
 | ||
| 
 | ||
|         #region internal static Dictionary<VirtualKeys, Input.Key> KeyMap
 | ||
| 
 | ||
|         internal static Dictionary<VirtualKeys, Input.Key> KeyMap =
 | ||
|             new Dictionary<VirtualKeys, Input.Key>((int)VirtualKeys.Last);
 | ||
| 
 | ||
|         private static bool keyMapExists;
 | ||
|         /// <summary>
 | ||
|         /// Initializes the map between VirtualKeys and OpenTK.Key
 | ||
|         /// </summary>
 | ||
|         private static void InitKeyMap()
 | ||
|         {
 | ||
|             if (!keyMapExists)
 | ||
|             {
 | ||
|                 try
 | ||
|                 {
 | ||
|                     KeyMap.Add(VirtualKeys.ESCAPE, Input.Key.Escape);
 | ||
| 
 | ||
|                     // Function keys
 | ||
|                     for (int i = 0; i < 24; i++)
 | ||
|                     {
 | ||
|                         KeyMap.Add((VirtualKeys)((int)VirtualKeys.F1 + i), Input.Key.F1 + i);
 | ||
|                     }
 | ||
| 
 | ||
|                     // Number keys (0-9)
 | ||
|                     for (int i = 0; i <= 9; i++)
 | ||
|                     {
 | ||
|                         KeyMap.Add((VirtualKeys)(0x30 + i), Input.Key.Number0 + i);
 | ||
|                     }
 | ||
| 
 | ||
|                     // Letters (A-Z)
 | ||
|                     for (int i = 0; i < 26; i++)
 | ||
|                     {
 | ||
|                         KeyMap.Add((VirtualKeys)(0x41 + i), Input.Key.A + i);
 | ||
|                     }
 | ||
| 
 | ||
|                     KeyMap.Add(VirtualKeys.TAB, Input.Key.Tab);
 | ||
|                     KeyMap.Add(VirtualKeys.CAPITAL, Input.Key.CapsLock);
 | ||
|                     KeyMap.Add(VirtualKeys.LCONTROL, Input.Key.ControlLeft);
 | ||
|                     KeyMap.Add(VirtualKeys.LSHIFT, Input.Key.ShiftLeft);
 | ||
|                     KeyMap.Add(VirtualKeys.LWIN, Input.Key.WinLeft);
 | ||
|                     KeyMap.Add(VirtualKeys.LMENU, Input.Key.AltLeft);
 | ||
|                     KeyMap.Add(VirtualKeys.SPACE, Input.Key.Space);
 | ||
|                     KeyMap.Add(VirtualKeys.RMENU, Input.Key.AltRight);
 | ||
|                     KeyMap.Add(VirtualKeys.RWIN, Input.Key.WinRight);
 | ||
|                     KeyMap.Add(VirtualKeys.APPS, Input.Key.Menu);
 | ||
|                     KeyMap.Add(VirtualKeys.RCONTROL, Input.Key.ControlRight);
 | ||
|                     KeyMap.Add(VirtualKeys.RSHIFT, Input.Key.ShiftRight);
 | ||
|                     KeyMap.Add(VirtualKeys.RETURN, Input.Key.Enter);
 | ||
|                     KeyMap.Add(VirtualKeys.BACK, Input.Key.BackSpace);
 | ||
| 
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_1, Input.Key.Semicolon);      // Varies by keyboard, ;: on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_2, Input.Key.Slash);          // Varies by keyboard, /? on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_3, Input.Key.Tilde);          // Varies by keyboard, `~ on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_4, Input.Key.BracketLeft);    // Varies by keyboard, [{ on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_5, Input.Key.BackSlash);      // Varies by keyboard, \| on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_6, Input.Key.BracketRight);   // Varies by keyboard, ]} on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_7, Input.Key.Quote);          // Varies by keyboard, '" on Win2K/US
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_PLUS, Input.Key.Plus);        // Invariant: +
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_COMMA, Input.Key.Comma);      // Invariant: ,
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_MINUS, Input.Key.Minus);      // Invariant: -
 | ||
|                     KeyMap.Add(VirtualKeys.OEM_PERIOD, Input.Key.Period);    // Invariant: .
 | ||
| 
 | ||
|                     KeyMap.Add(VirtualKeys.HOME, Input.Key.Home);
 | ||
|                     KeyMap.Add(VirtualKeys.END, Input.Key.End);
 | ||
|                     KeyMap.Add(VirtualKeys.DELETE, Input.Key.Delete);
 | ||
|                     KeyMap.Add(VirtualKeys.PRIOR, Input.Key.PageUp);
 | ||
|                     KeyMap.Add(VirtualKeys.NEXT, Input.Key.PageDown);
 | ||
|                     KeyMap.Add(VirtualKeys.PRINT, Input.Key.PrintScreen);
 | ||
|                     KeyMap.Add(VirtualKeys.PAUSE, Input.Key.Pause);
 | ||
|                     KeyMap.Add(VirtualKeys.NUMLOCK, Input.Key.NumLock);
 | ||
|                     
 | ||
|                     KeyMap.Add(VirtualKeys.SLEEP, Input.Key.Sleep);
 | ||
| 
 | ||
|                     // Keypad
 | ||
|                     for (int i = 0; i <= 9; i++)
 | ||
|                     {
 | ||
|                         KeyMap.Add((VirtualKeys)((int)VirtualKeys.NUMPAD0 + i), Input.Key.Keypad0 + i);
 | ||
| 
 | ||
|                     }
 | ||
|                     KeyMap.Add(VirtualKeys.DECIMAL, Input.Key.KeypadDecimal);
 | ||
|                     KeyMap.Add(VirtualKeys.ADD, Input.Key.KeypadAdd);
 | ||
|                     KeyMap.Add(VirtualKeys.SUBTRACT, Input.Key.KeypadSubtract);
 | ||
|                     KeyMap.Add(VirtualKeys.DIVIDE, Input.Key.KeypadDivide);
 | ||
|                     KeyMap.Add(VirtualKeys.MULTIPLY, Input.Key.KeypadMultiply);
 | ||
| 
 | ||
|                     // Navigation
 | ||
|                     KeyMap.Add(VirtualKeys.UP, Input.Key.Up);
 | ||
|                     KeyMap.Add(VirtualKeys.DOWN, Input.Key.Down);
 | ||
|                     KeyMap.Add(VirtualKeys.LEFT, Input.Key.Left);
 | ||
|                     KeyMap.Add(VirtualKeys.RIGHT, Input.Key.Right);
 | ||
|                 }
 | ||
|                 catch (ArgumentException e)
 | ||
|                 {
 | ||
|                     Debug.Print("Exception while creating keymap: '{0}'.", e.ToString());
 | ||
|                     System.Windows.Forms.MessageBox.Show(
 | ||
|                         String.Format("Exception while creating keymap: '{0}'.", e.ToString()));
 | ||
|                 }
 | ||
|                 finally
 | ||
|                 {
 | ||
|                     keyMapExists = true;
 | ||
|                 }
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region --- Constructors ---
 | ||
| 
 | ||
|         internal WinRawKeyboard()
 | ||
|             : this(IntPtr.Zero)
 | ||
|         {
 | ||
|         }
 | ||
| 
 | ||
|         internal WinRawKeyboard(IntPtr windowHandle)
 | ||
|         {
 | ||
|             Debug.WriteLine("Initializing keyboard driver.");
 | ||
|             Debug.Indent();
 | ||
| 
 | ||
|             this.windowHandle = windowHandle;
 | ||
|             InitKeyMap();
 | ||
| 
 | ||
|             UpdateKeyboardList();
 | ||
| 
 | ||
|             Debug.Unindent();
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region internal static void UpdateKeyboardList()
 | ||
| 
 | ||
|         internal void UpdateKeyboardList()
 | ||
|         {
 | ||
|             int count = WinRawInput.DeviceCount;
 | ||
|             RawInputDeviceList[] ridl = new RawInputDeviceList[count];
 | ||
|             for (int i = 0; i < count; i++)
 | ||
|                 ridl[i] = new RawInputDeviceList();
 | ||
|             API.GetRawInputDeviceList(ridl, ref count, API.RawInputDeviceListSize);
 | ||
| 
 | ||
|             // Discover keyboard devices:
 | ||
|             for (int i = 0; i < count; i++)
 | ||
|             {
 | ||
|                 uint size = 0;
 | ||
|                 API.GetRawInputDeviceInfo(ridl[i].Device, RawInputDeviceInfoEnum.DEVICENAME, IntPtr.Zero, ref size);
 | ||
|                 IntPtr name_ptr = Marshal.AllocHGlobal((IntPtr)size);
 | ||
|                 API.GetRawInputDeviceInfo(ridl[i].Device, RawInputDeviceInfoEnum.DEVICENAME, name_ptr, ref size);
 | ||
|                 string name = Marshal.PtrToStringAnsi(name_ptr);
 | ||
|                 Marshal.FreeHGlobal(name_ptr);
 | ||
|                 if (name.ToLower().Contains("root"))
 | ||
|                 {
 | ||
|                     // This is a terminal services devices, skip it.
 | ||
|                     continue;
 | ||
|                 }
 | ||
|                 else if (ridl[i].Type == RawInputDeviceType.KEYBOARD || ridl[i].Type == RawInputDeviceType.HID)
 | ||
|                 {
 | ||
|                     //It's a keyboard – or a USB device that could be a keyboard
 | ||
| 
 | ||
|                     // remove the \??\
 | ||
|                     name = name.Substring(4);
 | ||
| 
 | ||
|                     string[] split = name.Split('#');
 | ||
| 
 | ||
|                     string id_01 = split[0];    // ACPI (Class code)
 | ||
|                     string id_02 = split[1];    // PNP0303 (SubClass code)
 | ||
|                     string id_03 = split[2];    // 3&13c0b0c5&0 (Protocol code)
 | ||
|                     // The final part is the class GUID and is not needed here
 | ||
| 
 | ||
| 
 | ||
|                     string findme = string.Format(
 | ||
|                         @"System\CurrentControlSet\Enum\{0}\{1}\{2}",
 | ||
|                         id_01, id_02, id_03);
 | ||
| 
 | ||
|                     RegistryKey regkey = Registry.LocalMachine.OpenSubKey(findme);
 | ||
| 
 | ||
|                     string deviceDesc =
 | ||
|                         (string)regkey.GetValue("DeviceDesc");
 | ||
|                     string deviceClass =
 | ||
|                         (string)regkey.GetValue("Class");
 | ||
|                     if (!String.IsNullOrEmpty(deviceClass) && deviceClass.ToLower().Equals("keyboard"))
 | ||
|                     {
 | ||
|                         Keyboard kb = new Keyboard();
 | ||
|                         kb.Description = deviceDesc;
 | ||
| 
 | ||
|                         // Register the keyboard:
 | ||
|                         RawInputDeviceInfo info = new RawInputDeviceInfo();
 | ||
|                         int devInfoSize = API.RawInputDeviceInfoSize;
 | ||
|                         API.GetRawInputDeviceInfo(ridl[i].Device, RawInputDeviceInfoEnum.DEVICEINFO,
 | ||
|                                 info, ref devInfoSize);
 | ||
| 
 | ||
|                         kb.NumberOfLeds = info.Device.Keyboard.NumberOfIndicators;
 | ||
|                         kb.NumberOfFunctionKeys = info.Device.Keyboard.NumberOfFunctionKeys;
 | ||
|                         kb.NumberOfKeys = info.Device.Keyboard.NumberOfKeysTotal;
 | ||
|                         kb.DeviceID = (info.Device.Keyboard.Type << 32) + info.Device.Keyboard.SubType;
 | ||
| 
 | ||
|                         if (!keyboards.Contains(kb))
 | ||
|                         {
 | ||
|                             this.RegisterKeyboardDevice(kb);
 | ||
|                             keyboards.Add(kb);
 | ||
|                         }
 | ||
|                     }
 | ||
|                 }
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region internal void RegisterKeyboardDevice(Keyboard kb)
 | ||
| 
 | ||
|         internal void RegisterKeyboardDevice(Keyboard kb)
 | ||
|         {
 | ||
|             RawInputDevice[] rid = new RawInputDevice[1];
 | ||
|             // Keyboard is 1/6 (page/id). See http://www.microsoft.com/whdc/device/input/HID_HWID.mspx
 | ||
|             rid[0] = new RawInputDevice();
 | ||
|             rid[0].UsagePage = 1;
 | ||
|             rid[0].Usage = 6;
 | ||
|             rid[0].Flags = RawInputDeviceFlags.INPUTSINK;
 | ||
|             rid[0].Target = windowHandle;
 | ||
| 
 | ||
|             if (!API.RegisterRawInputDevices(rid, 1, API.RawInputDeviceSize))
 | ||
|             {
 | ||
|                 throw new ApplicationException(
 | ||
|                     String.Format(
 | ||
|                         "Raw input registration failed with error: {0}. Device: {1}",
 | ||
|                         Marshal.GetLastWin32Error(),
 | ||
|                         rid[0].ToString())
 | ||
|                 );
 | ||
|             }
 | ||
|             else
 | ||
|             {
 | ||
|                 Debug.Print("Registered keyboard {0}", kb.ToString());
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region internal bool ProcessKeyboardEvent(API.RawInput rin)
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Processes raw input events.
 | ||
|         /// </summary>
 | ||
|         /// <param name="rin"></param>
 | ||
|         /// <returns></returns>
 | ||
|         internal bool ProcessKeyboardEvent(RawInput rin)
 | ||
|         {
 | ||
|             //Keyboard key = keyboards[0];
 | ||
|             //rin.Header.Device;
 | ||
|             switch (rin.Header.Type)
 | ||
|             {
 | ||
|                 case RawInputDeviceType.KEYBOARD:
 | ||
|                     bool pressed =
 | ||
|                         rin.Data.Keyboard.Message == (int)WindowMessage.KEYDOWN ||
 | ||
|                         rin.Data.Keyboard.Message == (int)WindowMessage.SYSKEYDOWN;
 | ||
| 
 | ||
|                     // Generic control, shift, alt keys may be sent instead of left/right.
 | ||
|                     // It seems you have to explicitly register left/right events.
 | ||
|                     switch (rin.Data.Keyboard.VKey)
 | ||
|                     {
 | ||
|                         case VirtualKeys.SHIFT:
 | ||
|                             keyboards[0][Input.Key.ShiftLeft] = keyboards[0][Input.Key.ShiftRight] = pressed;
 | ||
|                             return true;
 | ||
| 
 | ||
|                         case VirtualKeys.CONTROL:
 | ||
|                             keyboards[0][Input.Key.ControlLeft] = keyboards[0][Input.Key.ControlRight] = pressed;
 | ||
|                             return true;
 | ||
| 
 | ||
|                         case VirtualKeys.MENU:
 | ||
|                             keyboards[0][Input.Key.AltLeft] = keyboards[0][Input.Key.AltRight] = pressed;
 | ||
|                             return true;
 | ||
| 
 | ||
|                         default:
 | ||
|                             if (!WinRawKeyboard.KeyMap.ContainsKey(rin.Data.Keyboard.VKey))
 | ||
|                             {
 | ||
|                                 Debug.Print("Virtual key {0} not mapped.", rin.Data.Keyboard.VKey);
 | ||
|                             }
 | ||
|                             else
 | ||
|                             {
 | ||
|                                 keyboards[0][WinRawKeyboard.KeyMap[rin.Data.Keyboard.VKey]] = pressed;
 | ||
|                             }
 | ||
|                             return false;
 | ||
|                     }
 | ||
| 
 | ||
|                 default:
 | ||
|                     throw new ApplicationException("Windows raw keyboard driver received invalid data.");
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region --- IInputDevice Members ---
 | ||
| 
 | ||
|         public string Description
 | ||
|         {
 | ||
|             get { throw new Exception("The method or operation is not implemented."); }
 | ||
|         }
 | ||
| 
 | ||
|         public Input.InputDeviceType DeviceType
 | ||
|         {
 | ||
|             get { return Input.InputDeviceType.Keyboard; }
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region --- IKeyboardDriver Members ---
 | ||
| 
 | ||
|         public IList<Keyboard> Keyboard
 | ||
|         {
 | ||
|             get { return keyboards; }
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
|         #region --- IDisposable Members ---
 | ||
| 
 | ||
|         private bool disposed;
 | ||
| 
 | ||
|         public void Dispose()
 | ||
|         {
 | ||
|             Dispose(true);
 | ||
|             GC.SuppressFinalize(this);
 | ||
|         }
 | ||
| 
 | ||
|         private void Dispose(bool manual)
 | ||
|         {
 | ||
|             if (!disposed)
 | ||
|             {
 | ||
|                 if (manual)
 | ||
|                 {
 | ||
|                     keyboards.Clear();
 | ||
|                 }
 | ||
|                 disposed = true;
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         ~WinRawKeyboard()
 | ||
|         {
 | ||
|             Dispose(false);
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
|     }
 | ||
| }
 |