2013-12-24 13:22:03 +00:00
|
|
|
|
#region License
|
|
|
|
|
//
|
|
|
|
|
// MappedGamePadDriver.cs
|
|
|
|
|
//
|
|
|
|
|
// Author:
|
|
|
|
|
// Stefanos A. <stapostol@gmail.com>
|
|
|
|
|
//
|
|
|
|
|
// Copyright (c) 2006-2013 Stefanos Apostolopoulos
|
|
|
|
|
//
|
|
|
|
|
// 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.Text;
|
|
|
|
|
using OpenTK.Input;
|
|
|
|
|
|
|
|
|
|
namespace OpenTK.Platform
|
|
|
|
|
{
|
|
|
|
|
/// \internal
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Implements IGamePadDriver using OpenTK.Input.Joystick
|
|
|
|
|
/// and a gamepad-specific axis/button mapping.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// <para>
|
|
|
|
|
/// This class supports OpenTK and is not meant to be accessed by user code.
|
|
|
|
|
/// </para>
|
|
|
|
|
/// <para>
|
|
|
|
|
/// To support gamepads on platforms that do not offer a gamepad-optimized API,
|
|
|
|
|
/// we need to use the generic OpenTK.Input.Joystick and implement a custom
|
|
|
|
|
/// mapping scheme to provide a stable mapping to OpenTK.Input.GamePad. This
|
|
|
|
|
/// class implements this mapping scheme.
|
|
|
|
|
/// </para>
|
|
|
|
|
/// </remarks>
|
|
|
|
|
class MappedGamePadDriver : IGamePadDriver
|
|
|
|
|
{
|
2014-01-02 01:42:51 +00:00
|
|
|
|
readonly GamePadConfigurationDatabase database =
|
|
|
|
|
new GamePadConfigurationDatabase();
|
2014-01-02 00:38:12 +00:00
|
|
|
|
readonly Dictionary<Guid, GamePadConfiguration> configurations =
|
|
|
|
|
new Dictionary<Guid, GamePadConfiguration>();
|
2014-01-02 00:13:20 +00:00
|
|
|
|
|
2013-12-24 13:22:03 +00:00
|
|
|
|
public GamePadState GetState(int index)
|
|
|
|
|
{
|
|
|
|
|
JoystickState joy = Joystick.GetState(index);
|
|
|
|
|
GamePadState pad = new GamePadState();
|
2014-01-02 17:37:53 +00:00
|
|
|
|
|
2013-12-24 13:22:03 +00:00
|
|
|
|
if (joy.IsConnected)
|
|
|
|
|
{
|
2014-01-02 17:37:53 +00:00
|
|
|
|
pad.SetConnected(true);
|
2014-01-02 18:24:15 +00:00
|
|
|
|
pad.SetPacketNumber(joy.PacketNumber);
|
|
|
|
|
|
2014-01-02 17:37:53 +00:00
|
|
|
|
GamePadConfiguration configuration = GetConfiguration(Joystick.GetGuid(index));
|
|
|
|
|
|
|
|
|
|
foreach (GamePadConfigurationItem map in configuration)
|
|
|
|
|
{
|
|
|
|
|
switch (map.Source.Type)
|
|
|
|
|
{
|
|
|
|
|
case ConfigurationType.Axis:
|
|
|
|
|
{
|
|
|
|
|
// JoystickAxis -> Buttons/GamePadAxes mapping
|
|
|
|
|
JoystickAxis source_axis = map.Source.Axis;
|
|
|
|
|
short value = joy.GetAxisRaw(source_axis);
|
|
|
|
|
|
|
|
|
|
switch (map.Target.Type)
|
|
|
|
|
{
|
|
|
|
|
case ConfigurationType.Axis:
|
2014-01-29 09:39:18 +00:00
|
|
|
|
pad.SetAxis(map.Target.Axis, value);
|
2014-01-02 17:37:53 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ConfigurationType.Button:
|
2014-01-29 09:39:18 +00:00
|
|
|
|
// Todo: if SDL2 GameController config is ever updated to
|
|
|
|
|
// distinguish between negative/positive axes, then remove
|
|
|
|
|
// Math.Abs below.
|
|
|
|
|
// Button is considered press when the axis is >= 0.5 from center
|
|
|
|
|
pad.SetButton(map.Target.Button, Math.Abs(value) >= short.MaxValue >> 1);
|
2014-01-02 17:37:53 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
2014-01-02 00:13:20 +00:00
|
|
|
|
|
2014-01-02 17:37:53 +00:00
|
|
|
|
case ConfigurationType.Button:
|
|
|
|
|
{
|
|
|
|
|
// JoystickButton -> Buttons/GamePadAxes mapping
|
|
|
|
|
JoystickButton source_button = map.Source.Button;
|
|
|
|
|
bool pressed = joy.GetButton(source_button) == ButtonState.Pressed;
|
2014-01-02 00:13:20 +00:00
|
|
|
|
|
2014-01-02 17:37:53 +00:00
|
|
|
|
switch (map.Target.Type)
|
|
|
|
|
{
|
|
|
|
|
case ConfigurationType.Axis:
|
2014-01-29 09:39:18 +00:00
|
|
|
|
// Todo: if SDL2 GameController config is ever updated to
|
2014-07-25 21:26:03 +00:00
|
|
|
|
// distinguish between negative/positive axes, then update
|
|
|
|
|
// the following line to support both.
|
2014-09-02 22:11:00 +00:00
|
|
|
|
short value = pressed ?
|
|
|
|
|
short.MaxValue :
|
|
|
|
|
map.Target.Axis == GamePadAxes.LeftTrigger || map.Target.Axis == GamePadAxes.RightTrigger ?
|
|
|
|
|
short.MinValue :
|
|
|
|
|
(short)0;
|
|
|
|
|
pad.SetAxis(map.Target.Axis, value);
|
2014-07-25 21:26:03 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ConfigurationType.Button:
|
|
|
|
|
pad.SetButton(map.Target.Button, pressed);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ConfigurationType.Hat:
|
|
|
|
|
{
|
|
|
|
|
// JoystickHat -> Buttons/GamePadAxes mapping
|
|
|
|
|
JoystickHat source_hat = map.Source.Hat;
|
2014-07-29 21:21:37 +00:00
|
|
|
|
JoystickHatState state = joy.GetHat(source_hat);
|
2014-07-27 13:54:36 +00:00
|
|
|
|
|
2014-07-29 21:21:37 +00:00
|
|
|
|
bool pressed = false;
|
|
|
|
|
switch (map.Source.HatPosition)
|
|
|
|
|
{
|
|
|
|
|
case HatPosition.Down:
|
|
|
|
|
pressed = state.IsDown;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case HatPosition.Up:
|
|
|
|
|
pressed = state.IsUp;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case HatPosition.Left:
|
|
|
|
|
pressed = state.IsLeft;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case HatPosition.Right:
|
|
|
|
|
pressed = state.IsRight;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2014-07-25 21:26:03 +00:00
|
|
|
|
|
|
|
|
|
switch (map.Target.Type)
|
|
|
|
|
{
|
|
|
|
|
case ConfigurationType.Axis:
|
|
|
|
|
// Todo: if SDL2 GameController config is ever updated to
|
2014-01-29 09:39:18 +00:00
|
|
|
|
// distinguish between negative/positive axes, then update
|
|
|
|
|
// the following line to support both.
|
2014-09-02 22:11:00 +00:00
|
|
|
|
short value = pressed ?
|
|
|
|
|
short.MaxValue :
|
|
|
|
|
map.Target.Axis == GamePadAxes.LeftTrigger || map.Target.Axis == GamePadAxes.RightTrigger ?
|
|
|
|
|
short.MinValue :
|
|
|
|
|
(short)0;
|
|
|
|
|
pad.SetAxis(map.Target.Axis, value);
|
2014-01-02 17:37:53 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ConfigurationType.Button:
|
2014-01-29 09:39:18 +00:00
|
|
|
|
pad.SetButton(map.Target.Button, pressed);
|
2014-01-02 17:37:53 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-12-24 13:22:03 +00:00
|
|
|
|
}
|
2014-01-02 17:37:53 +00:00
|
|
|
|
|
2013-12-24 13:22:03 +00:00
|
|
|
|
return pad;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public GamePadCapabilities GetCapabilities(int index)
|
|
|
|
|
{
|
|
|
|
|
JoystickCapabilities joy = Joystick.GetCapabilities(index);
|
2014-01-02 00:13:20 +00:00
|
|
|
|
GamePadCapabilities pad;
|
2013-12-24 13:22:03 +00:00
|
|
|
|
if (joy.IsConnected)
|
|
|
|
|
{
|
2014-01-02 17:37:53 +00:00
|
|
|
|
GamePadConfiguration configuration = GetConfiguration(Joystick.GetGuid(index));
|
|
|
|
|
GamePadAxes mapped_axes = 0;
|
|
|
|
|
Buttons mapped_buttons = 0;
|
|
|
|
|
|
|
|
|
|
foreach (GamePadConfigurationItem map in configuration)
|
|
|
|
|
{
|
|
|
|
|
switch (map.Target.Type)
|
|
|
|
|
{
|
|
|
|
|
case ConfigurationType.Axis:
|
|
|
|
|
mapped_axes |= map.Target.Axis;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ConfigurationType.Button:
|
|
|
|
|
mapped_buttons |= map.Target.Button;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-02 00:13:20 +00:00
|
|
|
|
pad = new GamePadCapabilities(
|
|
|
|
|
GamePadType.GamePad, // Todo: detect different types
|
2014-01-02 17:37:53 +00:00
|
|
|
|
mapped_axes,
|
|
|
|
|
mapped_buttons,
|
2014-09-12 06:36:52 +00:00
|
|
|
|
joy.IsConnected,
|
|
|
|
|
configuration.Name != GamePadConfigurationDatabase.UnmappedName);
|
2014-01-02 00:13:20 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
pad = new GamePadCapabilities();
|
2013-12-24 13:22:03 +00:00
|
|
|
|
}
|
|
|
|
|
return pad;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetName(int index)
|
|
|
|
|
{
|
2014-01-02 00:34:21 +00:00
|
|
|
|
JoystickCapabilities joy = Joystick.GetCapabilities(index);
|
|
|
|
|
string name = String.Empty;
|
|
|
|
|
if (joy.IsConnected)
|
|
|
|
|
{
|
2014-01-02 00:38:12 +00:00
|
|
|
|
GamePadConfiguration map = GetConfiguration(Joystick.GetGuid(index));
|
2014-01-02 00:34:21 +00:00
|
|
|
|
name = map.Name;
|
|
|
|
|
}
|
|
|
|
|
return name;
|
2013-12-24 13:22:03 +00:00
|
|
|
|
}
|
2014-01-02 00:13:20 +00:00
|
|
|
|
|
2014-01-02 18:52:00 +00:00
|
|
|
|
public bool SetVibration(int index, float left, float right)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-02 00:13:20 +00:00
|
|
|
|
#region Private Members
|
|
|
|
|
|
2014-01-02 00:38:12 +00:00
|
|
|
|
GamePadConfiguration GetConfiguration(Guid guid)
|
2014-01-02 00:13:20 +00:00
|
|
|
|
{
|
|
|
|
|
if (!configurations.ContainsKey(guid))
|
|
|
|
|
{
|
2014-01-02 01:42:51 +00:00
|
|
|
|
string config = database[guid];
|
|
|
|
|
GamePadConfiguration map = new GamePadConfiguration(config);
|
2014-01-02 00:13:20 +00:00
|
|
|
|
configurations.Add(guid, map);
|
|
|
|
|
}
|
|
|
|
|
return configurations[guid];
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-02 17:37:53 +00:00
|
|
|
|
bool IsMapped(GamePadConfigurationSource item)
|
2014-01-02 00:13:20 +00:00
|
|
|
|
{
|
2014-01-02 00:38:12 +00:00
|
|
|
|
return item.Type != ConfigurationType.Unmapped;
|
2014-01-02 00:13:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
2013-12-24 13:22:03 +00:00
|
|
|
|
}
|
|
|
|
|
}
|