#region License // // The Open Toolkit Library License // // Copyright (c) 2006 - 2010 the Open Toolkit library. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights to // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of // the Software, and to permit persons to whom the Software is furnished to do // so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR // OTHER DEALINGS IN THE SOFTWARE. // #endregion using System; using System.Collections.Generic; using System.Diagnostics; using System.Runtime.InteropServices; using OpenTK.Input; namespace OpenTK.Platform.X11 { // Todo: multi-mouse support. Right now we aggregate all data into a single mouse device. // This should be easy: just read the device id and route the data to the correct device. sealed class XI2Mouse : IMouseDriver2 { List mice = new List(); Dictionary rawids = new Dictionary(); // maps raw ids to mouse ids internal readonly X11WindowInfo window; internal static int XIOpCode { get; private set; } static bool supported; static readonly Functions.EventPredicate PredicateImpl = IsEventValid; readonly IntPtr Predicate = Marshal.GetFunctionPointerForDelegate(PredicateImpl); // Store information on a mouse warp event, so it can be ignored. struct MouseWarp : IEquatable { public MouseWarp(double x, double y) { X = x; Y = y; } double X, Y; public bool Equals(MouseWarp warp) { return X == warp.X && Y == warp.Y; } } MouseWarp? mouse_warp_event; int mouse_warp_event_count; public XI2Mouse() { Debug.WriteLine("Using XI2Mouse."); using (new XLock(API.DefaultDisplay)) { window = new X11WindowInfo(); window.Display = API.DefaultDisplay; window.Screen = Functions.XDefaultScreen(window.Display); window.RootWindow = Functions.XRootWindow(window.Display, window.Screen); window.Handle = window.RootWindow; } if (!IsSupported(window.Display)) throw new NotSupportedException("XInput2 not supported."); using (XIEventMask mask = new XIEventMask(1, XIEventMasks.RawButtonPressMask | XIEventMasks.RawButtonReleaseMask | XIEventMasks.RawMotionMask)) { Functions.XISelectEvents(window.Display, window.Handle, mask); Functions.XISelectEvents(window.Display, window.RootWindow, mask); } } // Checks whether XInput2 is supported on the specified display. // If a display is not specified, the default display is used. internal static bool IsSupported(IntPtr display) { if (display == IntPtr.Zero) display = API.DefaultDisplay; using (new XLock(display)) { int major, ev, error; if (Functions.XQueryExtension(display, "XInputExtension", out major, out ev, out error) == 0) { return false; } XIOpCode = major; } return true; } #region IMouseDriver2 Members public MouseState GetState() { ProcessEvents(); MouseState master = new MouseState(); foreach (MouseState ms in mice) { master.MergeBits(ms); } return master; } public MouseState GetState(int index) { ProcessEvents(); if (mice.Count > index) return mice[index]; else return new MouseState(); } public void SetPosition(double x, double y) { using (new XLock(window.Display)) { Functions.XWarpPointer(window.Display, IntPtr.Zero, window.RootWindow, 0, 0, 0, 0, (int)x, (int)y); // Mark the expected warp-event so it can be ignored. if (mouse_warp_event == null) mouse_warp_event_count = 0; mouse_warp_event_count++; mouse_warp_event = new MouseWarp((int)x, (int)y); } ProcessEvents(); } #endregion bool CheckMouseWarp(double x, double y) { // Check if a mouse warp with the specified destination exists. bool is_warp = mouse_warp_event.HasValue && mouse_warp_event.Value.Equals(new MouseWarp((int)x, (int)y)); if (is_warp && --mouse_warp_event_count <= 0) mouse_warp_event = null; return is_warp; } void ProcessEvents() { while (true) { XEvent e = new XEvent(); XGenericEventCookie cookie; using (new XLock(window.Display)) { if (!Functions.XCheckIfEvent(window.Display, ref e, Predicate, new IntPtr(XIOpCode))) return; cookie = e.GenericEventCookie; if (Functions.XGetEventData(window.Display, ref cookie) != 0) { XIRawEvent raw = (XIRawEvent) Marshal.PtrToStructure(cookie.data, typeof(XIRawEvent)); if (!rawids.ContainsKey(raw.deviceid)) { mice.Add(new MouseState()); rawids.Add(raw.deviceid, mice.Count - 1); } MouseState state = mice[rawids[raw.deviceid]]; switch (raw.evtype) { case XIEventType.RawMotion: double x = 0, y = 0; if (IsBitSet(raw.valuators.mask, 0)) { x = BitConverter.Int64BitsToDouble(Marshal.ReadInt64(raw.raw_values, 0)); } if (IsBitSet(raw.valuators.mask, 1)) { y = BitConverter.Int64BitsToDouble(Marshal.ReadInt64(raw.raw_values, 8)); } if (!CheckMouseWarp(x, y)) { state.X += (int)x; state.Y += (int)y; } break; case XIEventType.RawButtonPress: switch (raw.detail) { case 1: state.EnableBit((int)MouseButton.Left); break; case 2: state.EnableBit((int)MouseButton.Middle); break; case 3: state.EnableBit((int)MouseButton.Right); break; case 4: state.SetScrollRelative(0, 1); break; case 5: state.SetScrollRelative(0, -1); break; case 6: state.EnableBit((int)MouseButton.Button1); break; case 7: state.EnableBit((int)MouseButton.Button2); break; case 8: state.EnableBit((int)MouseButton.Button3); break; case 9: state.EnableBit((int)MouseButton.Button4); break; case 10: state.EnableBit((int)MouseButton.Button5); break; case 11: state.EnableBit((int)MouseButton.Button6); break; case 12: state.EnableBit((int)MouseButton.Button7); break; case 13: state.EnableBit((int)MouseButton.Button8); break; case 14: state.EnableBit((int)MouseButton.Button9); break; } break; case XIEventType.RawButtonRelease: switch (raw.detail) { case 1: state.DisableBit((int)MouseButton.Left); break; case 2: state.DisableBit((int)MouseButton.Middle); break; case 3: state.DisableBit((int)MouseButton.Right); break; case 6: state.DisableBit((int)MouseButton.Button1); break; case 7: state.DisableBit((int)MouseButton.Button2); break; case 8: state.DisableBit((int)MouseButton.Button3); break; case 9: state.DisableBit((int)MouseButton.Button4); break; case 10: state.DisableBit((int)MouseButton.Button5); break; case 11: state.DisableBit((int)MouseButton.Button6); break; case 12: state.DisableBit((int)MouseButton.Button7); break; case 13: state.DisableBit((int)MouseButton.Button8); break; case 14: state.DisableBit((int)MouseButton.Button9); break; } break; } mice[rawids[raw.deviceid]] = state; } Functions.XFreeEventData(window.Display, ref cookie); } } } static bool IsEventValid(IntPtr display, ref XEvent e, IntPtr arg) { return e.GenericEventCookie.extension == arg.ToInt32() && (e.GenericEventCookie.evtype == (int)XIEventType.RawMotion || e.GenericEventCookie.evtype == (int)XIEventType.RawButtonPress || e.GenericEventCookie.evtype == (int)XIEventType.RawButtonRelease); } static bool IsBitSet(IntPtr mask, int bit) { unsafe { return (*((byte*)mask + (bit >> 3)) & (1 << (bit & 7))) != 0; } } } }