2007-07-27 03:37:12 +02:00
|
|
|
|
#region --- License ---
|
2007-08-03 02:14:31 +02:00
|
|
|
|
/* Copyright (c) 2006, 2007 Stefanos Apostolopoulos
|
2007-07-27 03:37:12 +02:00
|
|
|
|
* See license.txt for license info
|
|
|
|
|
*/
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region --- Using directives ---
|
|
|
|
|
|
|
|
|
|
using System;
|
2007-07-27 03:20:55 +02:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Runtime.InteropServices;
|
2007-08-03 02:14:31 +02:00
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.Windows.Forms;
|
|
|
|
|
using OpenTK.Input;
|
2007-07-27 03:20:55 +02:00
|
|
|
|
|
2007-07-27 03:37:12 +02:00
|
|
|
|
#endregion
|
|
|
|
|
|
2007-07-27 03:20:55 +02:00
|
|
|
|
namespace OpenTK.Platform.Windows
|
|
|
|
|
{
|
2007-08-05 11:03:22 +02:00
|
|
|
|
internal class WinRawInput : NativeWindow, IInputDriver
|
2007-07-27 03:20:55 +02:00
|
|
|
|
{
|
2007-08-03 02:14:31 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Input event data.
|
|
|
|
|
/// </summary>
|
2007-09-02 02:16:22 +02:00
|
|
|
|
private RawInput data = new RawInput();
|
2007-08-03 02:14:31 +02:00
|
|
|
|
/// <summary>
|
2007-08-05 15:42:31 +02:00
|
|
|
|
/// The total number of input devices connected to this system.
|
2007-08-03 02:14:31 +02:00
|
|
|
|
/// </summary>
|
2007-08-05 15:42:31 +02:00
|
|
|
|
private static int deviceCount;
|
2007-09-22 15:13:17 +02:00
|
|
|
|
int rawInputStructSize = API.RawInputSize;
|
2007-08-05 15:42:31 +02:00
|
|
|
|
|
2007-08-04 14:09:58 +02:00
|
|
|
|
private WinRawKeyboard keyboardDriver;
|
2007-09-21 22:34:07 +02:00
|
|
|
|
private WinRawMouse mouseDriver;
|
2007-08-03 02:14:31 +02:00
|
|
|
|
|
2007-08-05 15:42:31 +02:00
|
|
|
|
#region --- Constructors ---
|
|
|
|
|
|
2007-09-21 22:34:07 +02:00
|
|
|
|
internal WinRawInput(IWindowInfo parent)
|
2007-08-03 02:14:31 +02:00
|
|
|
|
{
|
2007-08-05 15:42:31 +02:00
|
|
|
|
Debug.WriteLine("Initalizing windows raw input driver.");
|
2007-08-04 14:09:58 +02:00
|
|
|
|
Debug.Indent();
|
2007-08-06 13:22:18 +02:00
|
|
|
|
|
|
|
|
|
AssignHandle(parent.Handle);
|
|
|
|
|
Debug.Print("Input window attached to parent {0}", parent);
|
2007-08-04 14:09:58 +02:00
|
|
|
|
keyboardDriver = new WinRawKeyboard(this.Handle);
|
2007-09-21 22:34:07 +02:00
|
|
|
|
mouseDriver = new WinRawMouse(this.Handle);
|
2007-08-04 14:09:58 +02:00
|
|
|
|
|
|
|
|
|
Debug.Unindent();
|
2007-09-22 15:13:17 +02:00
|
|
|
|
|
2007-09-22 16:59:33 +02:00
|
|
|
|
//AllocateBuffer();
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-08-05 15:42:31 +02:00
|
|
|
|
#endregion
|
2007-08-03 02:14:31 +02:00
|
|
|
|
|
2007-08-06 13:22:18 +02:00
|
|
|
|
#region internal static int DeviceCount
|
|
|
|
|
|
2007-08-05 02:09:42 +02:00
|
|
|
|
internal static int DeviceCount
|
2007-08-03 02:14:31 +02:00
|
|
|
|
{
|
2007-08-04 14:09:58 +02:00
|
|
|
|
get
|
|
|
|
|
{
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.GetRawInputDeviceList(null, ref deviceCount, API.RawInputDeviceListSize);
|
2007-08-04 14:09:58 +02:00
|
|
|
|
return deviceCount;
|
|
|
|
|
}
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
2007-08-05 15:42:31 +02:00
|
|
|
|
|
2007-08-06 13:22:18 +02:00
|
|
|
|
#endregion
|
|
|
|
|
|
2007-08-03 02:14:31 +02:00
|
|
|
|
#region protected override void WndProc(ref Message msg)
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Processes the input Windows Message, routing the data to the correct Keyboard, Mouse or HID.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="msg">The WM_INPUT message, containing the data on the input event.</param>
|
|
|
|
|
protected override void WndProc(ref Message msg)
|
2007-07-27 03:20:55 +02:00
|
|
|
|
{
|
2007-09-02 02:16:22 +02:00
|
|
|
|
switch ((WindowMessage)msg.Msg)
|
2007-07-27 03:20:55 +02:00
|
|
|
|
{
|
2007-09-02 02:16:22 +02:00
|
|
|
|
case WindowMessage.INPUT:
|
2007-09-22 15:13:17 +02:00
|
|
|
|
int size = 0;
|
2007-08-04 14:09:58 +02:00
|
|
|
|
// Get the size of the input data
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.GetRawInputData(msg.LParam, GetRawInputDataEnum.INPUT,
|
2007-08-04 14:09:58 +02:00
|
|
|
|
IntPtr.Zero, ref size, API.RawInputHeaderSize);
|
|
|
|
|
|
|
|
|
|
//if (data == null || API.RawInputSize < size)
|
|
|
|
|
//{
|
|
|
|
|
// throw new ApplicationException("Critical error when processing raw windows input.");
|
|
|
|
|
//}
|
2007-09-26 13:57:32 +02:00
|
|
|
|
if (size == Functions.GetRawInputData(msg.LParam, GetRawInputDataEnum.INPUT,
|
2007-08-04 14:09:58 +02:00
|
|
|
|
data, ref size, API.RawInputHeaderSize))
|
2007-08-03 02:14:31 +02:00
|
|
|
|
{
|
2007-08-04 14:09:58 +02:00
|
|
|
|
switch (data.Header.Type)
|
|
|
|
|
{
|
2007-09-02 02:16:22 +02:00
|
|
|
|
case RawInputDeviceType.KEYBOARD:
|
2007-09-21 22:34:07 +02:00
|
|
|
|
if (!keyboardDriver.ProcessKeyboardEvent(data))
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.DefRawInputProc(ref data, 1, (uint)API.RawInputHeaderSize);
|
2007-09-21 22:34:07 +02:00
|
|
|
|
return;
|
2007-08-04 14:09:58 +02:00
|
|
|
|
|
2007-09-02 02:16:22 +02:00
|
|
|
|
case RawInputDeviceType.MOUSE:
|
2007-09-21 22:34:07 +02:00
|
|
|
|
if (!mouseDriver.ProcessEvent(data))
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.DefRawInputProc(ref data, 1, (uint)API.RawInputHeaderSize);
|
2007-09-21 22:34:07 +02:00
|
|
|
|
return;
|
2007-08-04 14:09:58 +02:00
|
|
|
|
|
2007-09-02 02:16:22 +02:00
|
|
|
|
case RawInputDeviceType.HID:
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.DefRawInputProc(ref data, 1, (uint)API.RawInputHeaderSize);
|
2007-09-21 22:34:07 +02:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2007-08-04 14:09:58 +02:00
|
|
|
|
}
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
2007-08-04 14:09:58 +02:00
|
|
|
|
else
|
2007-08-03 02:14:31 +02:00
|
|
|
|
{
|
2007-08-04 14:09:58 +02:00
|
|
|
|
throw new ApplicationException(String.Format(
|
|
|
|
|
"GetRawInputData returned invalid data. Windows error {0}. Please file a bug at http://opentk.sourceforge.net",
|
|
|
|
|
Marshal.GetLastWin32Error()));
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
|
|
|
|
break;
|
2007-09-22 16:59:33 +02:00
|
|
|
|
|
2007-09-02 02:16:22 +02:00
|
|
|
|
case WindowMessage.DESTROY:
|
2007-09-22 15:13:17 +02:00
|
|
|
|
Debug.Print("Input window detached from parent {0}.", Handle);
|
|
|
|
|
ReleaseHandle();
|
|
|
|
|
break;
|
2007-08-03 02:14:31 +02:00
|
|
|
|
|
2007-09-02 02:16:22 +02:00
|
|
|
|
case WindowMessage.QUIT:
|
2007-09-22 15:13:17 +02:00
|
|
|
|
Debug.WriteLine("Input window quit.");
|
|
|
|
|
this.Dispose();
|
|
|
|
|
break;
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
2007-08-04 14:09:58 +02:00
|
|
|
|
|
|
|
|
|
base.WndProc(ref msg);
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region --- IInputDriver Members ---
|
|
|
|
|
|
2007-09-26 13:57:32 +02:00
|
|
|
|
public IList<KeyboardDevice> Keyboard
|
2007-08-03 02:14:31 +02:00
|
|
|
|
{
|
2007-08-04 15:28:16 +02:00
|
|
|
|
get { return keyboardDriver.Keyboard; }
|
|
|
|
|
}
|
|
|
|
|
|
2007-09-26 13:57:32 +02:00
|
|
|
|
public IList<MouseDevice> Mouse
|
2007-08-04 15:28:16 +02:00
|
|
|
|
{
|
2007-09-21 22:34:07 +02:00
|
|
|
|
get { return mouseDriver.Mouse; }
|
2007-08-03 02:14:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-09-22 15:13:17 +02:00
|
|
|
|
public void Poll()
|
2007-08-05 20:26:14 +02:00
|
|
|
|
{
|
2007-09-22 15:13:17 +02:00
|
|
|
|
return;
|
2007-11-06 14:36:47 +01:00
|
|
|
|
#if false
|
2007-09-22 15:13:17 +02:00
|
|
|
|
// We will do a buffered read for all input devices and route the RawInput structures
|
|
|
|
|
// to the correct 'ProcessData' handlers. First, we need to find out the size of the
|
|
|
|
|
// buffer to allocate for the structures. Then we allocate the buffer and read the
|
|
|
|
|
// structures, calling the correct handler for each one. Last, we free the allocated
|
|
|
|
|
// buffer.
|
2007-09-22 16:59:33 +02:00
|
|
|
|
int size = 0;
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.GetRawInputBuffer(IntPtr.Zero, ref size, API.RawInputHeaderSize);
|
2007-09-22 16:59:33 +02:00
|
|
|
|
size *= 256;
|
|
|
|
|
IntPtr rin_data = Marshal.AllocHGlobal(size);
|
|
|
|
|
|
2007-09-22 15:13:17 +02:00
|
|
|
|
while (true)
|
|
|
|
|
{
|
|
|
|
|
// Iterate reading all available RawInput structures and routing them to their respective
|
|
|
|
|
// handlers.
|
2007-09-26 13:57:32 +02:00
|
|
|
|
int num = Functions.GetRawInputBuffer(rin_data, ref size, API.RawInputHeaderSize);
|
2007-09-22 15:13:17 +02:00
|
|
|
|
if (num == 0)
|
2007-09-22 16:59:33 +02:00
|
|
|
|
break;
|
2007-09-22 15:13:17 +02:00
|
|
|
|
else if (num < 0)
|
|
|
|
|
{
|
|
|
|
|
/*int error = Marshal.GetLastWin32Error();
|
|
|
|
|
if (error == 122)
|
|
|
|
|
{
|
|
|
|
|
// Enlarge the buffer, it was too small.
|
|
|
|
|
AllocateBuffer();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
throw new ApplicationException(String.Format(
|
|
|
|
|
"GetRawInputBuffer failed with code: {0}", error));
|
|
|
|
|
}*/
|
|
|
|
|
Debug.Print("GetRawInputBuffer failed with code: {0}", Marshal.GetLastWin32Error());
|
|
|
|
|
//AllocateBuffer();
|
2007-09-22 16:59:33 +02:00
|
|
|
|
break;
|
2007-09-22 15:13:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-09-22 16:59:33 +02:00
|
|
|
|
RawInput[] rin_structs = new RawInput[num];
|
2007-09-22 15:13:17 +02:00
|
|
|
|
IntPtr next_rin = rin_data;
|
2007-09-22 16:59:33 +02:00
|
|
|
|
for (int i = 0; i < num; i++)
|
2007-09-22 15:13:17 +02:00
|
|
|
|
{
|
2007-09-22 16:59:33 +02:00
|
|
|
|
rin_structs[i] = (RawInput)Marshal.PtrToStructure(next_rin, typeof(RawInput));
|
|
|
|
|
|
|
|
|
|
switch (rin_structs[i].Header.Type)
|
|
|
|
|
{
|
|
|
|
|
case RawInputDeviceType.KEYBOARD:
|
|
|
|
|
keyboardDriver.ProcessKeyboardEvent(rin_structs[i]);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case RawInputDeviceType.MOUSE:
|
|
|
|
|
mouseDriver.ProcessEvent(rin_structs[i]);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2007-09-26 13:57:32 +02:00
|
|
|
|
|
|
|
|
|
next_rin = Functions.NextRawInputStructure(next_rin);
|
2007-09-22 15:13:17 +02:00
|
|
|
|
}
|
2007-09-26 13:57:32 +02:00
|
|
|
|
Functions.DefRawInputProc(rin_structs, num, (uint)API.RawInputHeaderSize);
|
2007-09-22 15:13:17 +02:00
|
|
|
|
}
|
2007-09-22 16:59:33 +02:00
|
|
|
|
|
|
|
|
|
Marshal.FreeHGlobal(rin_data);
|
2007-11-06 14:36:47 +01:00
|
|
|
|
#endif
|
2007-08-05 20:26:14 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-08-03 02:14:31 +02:00
|
|
|
|
#endregion
|
2007-08-05 11:03:22 +02:00
|
|
|
|
|
2007-08-22 02:30:16 +02:00
|
|
|
|
#region --- IDisposable Members ---
|
2007-08-05 11:03:22 +02:00
|
|
|
|
|
2007-08-22 02:30:16 +02:00
|
|
|
|
private bool disposed;
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
2007-08-05 11:03:22 +02:00
|
|
|
|
{
|
2007-08-22 02:30:16 +02:00
|
|
|
|
Dispose(true);
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Dispose(bool manual)
|
|
|
|
|
{
|
|
|
|
|
if (!disposed)
|
|
|
|
|
{
|
|
|
|
|
if (manual)
|
|
|
|
|
{
|
|
|
|
|
keyboardDriver.Dispose();
|
2007-09-02 02:16:22 +02:00
|
|
|
|
this.ReleaseHandle();
|
2007-08-22 02:30:16 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
disposed = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~WinRawInput()
|
|
|
|
|
{
|
|
|
|
|
Dispose(false);
|
2007-08-05 11:03:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
2007-07-27 03:20:55 +02:00
|
|
|
|
}
|
|
|
|
|
}
|