using System; using System.Collections.Generic; using System.Text; using OpenTK.Input; using System.Runtime.InteropServices; using System.Diagnostics; namespace OpenTK.Platform.X11 { /// /// Drives the Keyboard devices on X11. /// This class supports OpenTK, and is not intended for use by OpenTK programs. /// internal class X11Keyboard : IKeyboardDriver { /// /// The window receiving input events. /// WindowInfo window; private bool[] keys = new bool[(int)OpenTK.Input.Key.MaxKeys]; private List keyboards = new List(); private static Dictionary keymap = new Dictionary((int)Key.MaxKeys); /// /// The smallest and largest KeyCode supported by the X server. Queried through API.DisplayKeycodes() /// int firstKeyCode, lastKeyCode; /// /// The number of KeySyms for each KeyCode. /// int keysyms_per_keycode; IntPtr[] keysyms; private static bool keymapExists; #region private static void Initialize() private static void Initialize() { if (!keymapExists) { keymap.Add(XKey.Shift_L, Key.ShiftLeft); keymap.Add(XKey.Shift_R, Key.ShiftRight); keymap.Add(XKey.Alt_L, Key.AltLeft); keymap.Add(XKey.Alt_R, Key.AltRight); keymap.Add(XKey.Control_L, Key.ControlLeft); keymap.Add(XKey.Control_R, Key.ControlRight); keymap.Add(XKey.Super_L, Key.WinLeft); keymap.Add(XKey.Super_R, Key.WinRight); keymap.Add(XKey.Meta_L, Key.WinLeft); keymap.Add(XKey.Meta_R, Key.WinRight); keymap.Add(XKey.Menu, Key.Menu); keymap.Add(XKey.Tab, Key.Tab); keymap.Add(XKey.minus, Key.Minus); keymap.Add(XKey.plus, Key.Plus); keymap.Add(XKey.equal, Key.Plus); //keymap.Add keymap.Add(XKey.Caps_Lock, Key.CapsLock); keymap.Add(XKey.Num_Lock, Key.NumLock); for (int i = (int)XKey.F1; i < (int)XKey.F35; i++) { keymap.Add((XKey)i, (Key)((int)Key.F1 + (i - (int)XKey.F1))); } for (int i = (int)XKey.a; i < (int)XKey.z; i++) { keymap.Add((XKey)i, (Key)((int)Key.A + (i - (int)XKey.a))); } for (int i = (int)XKey.A; i < (int)XKey.Z; i++) { keymap.Add((XKey)i, (Key)((int)Key.A + (i - (int)XKey.A))); } for (int i = (int)XKey.Number0; i <= (int)XKey.Number9; i++) { keymap.Add((XKey)i, (Key)((int)Key.Number0 + (i - (int)XKey.Number0))); } for (int i = (int)XKey.KP_0; i <= (int)XKey.KP_9; i++) { keymap.Add((XKey)i, (Key)((int)Key.Keypad0 + (i - (int)XKey.KP_0))); } keymap.Add(XKey.Pause, Key.Pause); keymap.Add(XKey.Break, Key.Pause); keymap.Add(XKey.Scroll_Lock, Key.Pause); keymap.Add(XKey.Insert, Key.PrintScreen); keymap.Add(XKey.Print, Key.PrintScreen); keymap.Add(XKey.Sys_Req, Key.PrintScreen); keymap.Add(XKey.braceleft, Key.BracketLeft); keymap.Add(XKey.bracketleft, Key.BracketLeft); keymap.Add(XKey.braceright, Key.BracketRight); keymap.Add(XKey.bracketright, Key.BracketRight); keymap.Add(XKey.colon, Key.Semicolon); keymap.Add(XKey.semicolon, Key.Semicolon); keymap.Add(XKey.quoteright, Key.Quote); keymap.Add(XKey.quotedbl, Key.Quote); keymap.Add(XKey.comma, Key.Comma); keymap.Add(XKey.less, Key.Comma); keymap.Add(XKey.period, Key.Period); keymap.Add(XKey.greater, Key.Period); keymap.Add(XKey.slash, Key.Slash); keymap.Add(XKey.question, Key.Slash); keymapExists = true; } } #endregion internal X11Keyboard(WindowInfo window) { this.window = window; Initialize(); API.DisplayKeycodes(window.Display, ref firstKeyCode, ref lastKeyCode); Debug.Print("First keycode: {0}, last {1}", firstKeyCode, lastKeyCode); IntPtr keysym_ptr = API.GetKeyboardMapping(window.Display, (byte)firstKeyCode, lastKeyCode - firstKeyCode + 1, ref keysyms_per_keycode); Debug.Print("{0} keysyms per keycode.", keysyms_per_keycode); keysyms = new IntPtr[(lastKeyCode - firstKeyCode + 1) * keysyms_per_keycode]; Marshal.PtrToStructure(keysym_ptr, keysyms); //keysyms = (IntPtr[])Marshal.PtrToStructure(keysym_ptr, typeof(IntPtr[])); API.Free(keysym_ptr); Keyboard kb = new Keyboard(); kb.Description = "Default X11 keyboard"; kb.NumberOfKeys = lastKeyCode - firstKeyCode + 1; keyboards.Add(kb); } #region internal bool ProcessKeyboardEvent(X11.KeyEvent e) /// /// Processes X11 KeyEvents. /// /// The X11.KeyEvent to process /// True if the event was processed, false otherwise. internal bool ProcessKeyboardEvent(X11.XKeyEvent e) { //int keysym = keysyms[(e.keycode - firstKeyCode) * keysyms_per_keycode].ToInt32(); //int keysym2 = keysyms[(e.keycode - firstKeyCode) * keysyms_per_keycode].ToInt32(); bool pressed = e.type == XEventName.KeyPress; IntPtr keysym = API.LookupKeysym(ref e, 0); IntPtr keysym2 = API.LookupKeysym(ref e, 1); switch (keysym.ToInt64()) { default: if (keymap.ContainsKey((XKey)keysym)) { keyboards[0][keymap[(XKey)keysym]] = pressed; } else if (keymap.ContainsKey((XKey)keysym2)) { keyboards[0][keymap[(XKey)keysym2]] = pressed; } else { Debug.Print("KeyCode {0} (Keysym: {1}, {2}) not mapped.", e.keycode, (XKey)keysym, (XKey)keysym2); } return true; } return false; /*API.e.keycode switch (rin.Header.Type) { case API.RawInputDeviceType.KEYBOARD: bool pressed = rin.Data.Keyboard.Message == API.Constants.WM_KEYDOWN || rin.Data.Keyboard.Message == API.Constants.WM_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 API.VirtualKeys.SHIFT: keyboards[0][Input.Key.ShiftLeft] = keyboards[0][Input.Key.ShiftRight] = pressed; return false; case API.VirtualKeys.CONTROL: keyboards[0][Input.Key.ControlLeft] = keyboards[0][Input.Key.ControlRight] = pressed; return false; case API.VirtualKeys.MENU: keyboards[0][Input.Key.AltLeft] = keyboards[0][Input.Key.AltRight] = pressed; return false; default: if (!WinRawKeyboard.KeyMap.ContainsKey(rin.Data.Keyboard.VKey)) { Debug.Print("Virtual key {0} not mapped.", rin.Data.Keyboard.VKey); OpenTK.OpenGL.GL.ClearColor(1.0f, 0.3f, 0.3f, 0.0f); } else { keyboards[0][WinRawKeyboard.KeyMap[rin.Data.Keyboard.VKey]] = pressed; OpenTK.OpenGL.GL.ClearColor(0.0f, 0.0f, 0.0f, 0.0f); } break; } break; default: throw new ApplicationException("Windows raw keyboard driver received invalid data."); } return false;*/ } #endregion #region --- IKeyboardDriver Members --- public IList Keyboard { get { return keyboards; } } #endregion } }