mirror of https://github.com/adamdruppe/arsd.git
basic support for game controllers
This commit is contained in:
parent
79f2d9f4a0
commit
4d73f858fd
|
@ -0,0 +1,339 @@
|
||||||
|
/*
|
||||||
|
On Linux, I'll just use /dev/input/js*. It is easy and works with everything I care about.
|
||||||
|
|
||||||
|
On Windows, I'll support the mmsystem messages as far as I can, and XInput for more capabilities
|
||||||
|
of the XBox 360 controller. (The mmsystem should support my old PS1 controller and xbox is the
|
||||||
|
other one I have. I have PS3 controllers too which would be nice but since they require additional
|
||||||
|
drivers, meh.)
|
||||||
|
|
||||||
|
linux notes:
|
||||||
|
all basic input is available, no audio (I think), no force feedback (I think)
|
||||||
|
|
||||||
|
winmm notes:
|
||||||
|
the xbox 360 controller basically works and sends events to the window for the buttons,
|
||||||
|
left stick, and triggers. It doesn't send events for the right stick or dpad, but these
|
||||||
|
are available through joyGetPositionEx (the dpad is the POV hat and the right stick is
|
||||||
|
the other axes).
|
||||||
|
|
||||||
|
The triggers are considered a z-axis with the left one going negative and right going positive.
|
||||||
|
|
||||||
|
windows xinput notes:
|
||||||
|
all xbox 360 controller features are available via a polling api.
|
||||||
|
|
||||||
|
it doesn't seem to support events. That's OK for games generally though, because we just
|
||||||
|
want to check state on each loop.
|
||||||
|
|
||||||
|
For non-games however, using the traditional message loop is probably easier.
|
||||||
|
|
||||||
|
XInput is only supported on newer operating systems (Vista I think),
|
||||||
|
so I'm going to dynamically load it all and fallback on the old one if
|
||||||
|
it fails.
|
||||||
|
*/
|
||||||
|
|
||||||
|
version(Windows) {
|
||||||
|
|
||||||
|
import core.sys.windows.windows;
|
||||||
|
|
||||||
|
alias MMRESULT = UINT;
|
||||||
|
|
||||||
|
struct JOYINFOEX {
|
||||||
|
DWORD dwSize;
|
||||||
|
DWORD dwFlags;
|
||||||
|
DWORD dwXpos;
|
||||||
|
DWORD dwYpos;
|
||||||
|
DWORD dwZpos;
|
||||||
|
DWORD dwRpos;
|
||||||
|
DWORD dwUpos;
|
||||||
|
DWORD dwVpos;
|
||||||
|
DWORD dwButtons;
|
||||||
|
DWORD dwButtonNumber;
|
||||||
|
DWORD dwPOV;
|
||||||
|
DWORD dwReserved1;
|
||||||
|
DWORD dwReserved2;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum : DWORD {
|
||||||
|
JOY_POVCENTERED = -1,
|
||||||
|
JOY_POVFORWARD = 0,
|
||||||
|
JOY_POVBACKWARD = 18000,
|
||||||
|
JOY_POVLEFT = 27000,
|
||||||
|
JOY_POVRIGHT = 9000
|
||||||
|
}
|
||||||
|
|
||||||
|
extern(Windows)
|
||||||
|
MMRESULT joySetCapture(HWND window, UINT stickId, UINT period, BOOL changed);
|
||||||
|
|
||||||
|
extern(Windows)
|
||||||
|
MMRESULT joyGetPosEx(UINT stickId, JOYINFOEX* pji);
|
||||||
|
|
||||||
|
extern(Windows)
|
||||||
|
MMRESULT joyReleaseCapture(UINT stickId);
|
||||||
|
|
||||||
|
// SEE ALSO:
|
||||||
|
// http://msdn.microsoft.com/en-us/library/windows/desktop/dd757105%28v=vs.85%29.aspx
|
||||||
|
|
||||||
|
// Windows also provides joyGetThreshold, joySetThreshold
|
||||||
|
|
||||||
|
// there's also JOY2 messages
|
||||||
|
enum MM_JOY1MOVE = 0; // FIXME
|
||||||
|
enum MM_JOY1BUTTONDOWN = 0; // FIXME
|
||||||
|
enum MM_JOY1BUTTONUP = 0; // FIXME
|
||||||
|
|
||||||
|
pragma(lib, "winmm");
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
/*
|
||||||
|
auto window = new SimpleWindow(500, 500);
|
||||||
|
|
||||||
|
joySetCapture(window.impl.hwnd, 0, 0, false);
|
||||||
|
|
||||||
|
window.handleNativeEvent = (HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) {
|
||||||
|
import std.stdio;
|
||||||
|
writeln(msg, " ", wparam, " ", lparam);
|
||||||
|
return 1;
|
||||||
|
};
|
||||||
|
|
||||||
|
window.eventLoop(0);
|
||||||
|
|
||||||
|
joyReleaseCapture(0);
|
||||||
|
*/
|
||||||
|
|
||||||
|
import std.stdio;
|
||||||
|
|
||||||
|
WindowXInput x;
|
||||||
|
if(!x.loadDll()) {
|
||||||
|
writeln("Load DLL failed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
writeln("success");
|
||||||
|
|
||||||
|
assert(x.XInputSetState !is null);
|
||||||
|
assert(x.XInputGetState !is null);
|
||||||
|
|
||||||
|
XINPUT_STATE state;
|
||||||
|
|
||||||
|
XINPUT_VIBRATION vibration;
|
||||||
|
|
||||||
|
if(!x.XInputGetState(0, &state)) {
|
||||||
|
writeln("Player 1 detected");
|
||||||
|
} else return;
|
||||||
|
if(!x.XInputGetState(1, &state)) {
|
||||||
|
writeln("Player 2 detected");
|
||||||
|
} else writeln("Player 2 not found");
|
||||||
|
|
||||||
|
DWORD pn;
|
||||||
|
foreach(i; 0 .. 60) {
|
||||||
|
x.XInputGetState(0, &state);
|
||||||
|
if(pn != state.dwPacketNumber) {
|
||||||
|
writeln("c: ", state);
|
||||||
|
pn = state.dwPacketNumber;
|
||||||
|
}
|
||||||
|
Sleep(50);
|
||||||
|
if(i == 20) {
|
||||||
|
vibration.wLeftMotorSpeed = WORD.max;
|
||||||
|
vibration.wRightMotorSpeed = WORD.max;
|
||||||
|
x.XInputSetState(0, &vibration);
|
||||||
|
vibration = XINPUT_VIBRATION.init;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(i == 40)
|
||||||
|
x.XInputSetState(0, &vibration);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct XINPUT_GAMEPAD {
|
||||||
|
WORD wButtons;
|
||||||
|
BYTE bLeftTrigger;
|
||||||
|
BYTE bRightTrigger;
|
||||||
|
SHORT sThumbLX;
|
||||||
|
SHORT sThumbLY;
|
||||||
|
SHORT sThumbRX;
|
||||||
|
SHORT sThumbRY;
|
||||||
|
}
|
||||||
|
|
||||||
|
// enum XInputGamepadButtons {
|
||||||
|
// It is a bitmask of these
|
||||||
|
enum XINPUT_GAMEPAD_DPAD_UP = 0x0001;
|
||||||
|
enum XINPUT_GAMEPAD_DPAD_DOWN = 0x0002;
|
||||||
|
enum XINPUT_GAMEPAD_DPAD_LEFT = 0x0004;
|
||||||
|
enum XINPUT_GAMEPAD_DPAD_RIGHT = 0x0008;
|
||||||
|
enum XINPUT_GAMEPAD_START = 0x0010;
|
||||||
|
enum XINPUT_GAMEPAD_BACK = 0x0020;
|
||||||
|
enum XINPUT_GAMEPAD_LEFT_THUMB = 0x0040;
|
||||||
|
enum XINPUT_GAMEPAD_RIGHT_THUMB = 0x0080;
|
||||||
|
enum XINPUT_GAMEPAD_LEFT_SHOULDER = 0x0100;
|
||||||
|
enum XINPUT_GAMEPAD_RIGHT_SHOULDER = 0x0200;
|
||||||
|
enum XINPUT_GAMEPAD_A = 0x1000;
|
||||||
|
enum XINPUT_GAMEPAD_B = 0x2000;
|
||||||
|
enum XINPUT_GAMEPAD_X = 0x4000;
|
||||||
|
enum XINPUT_GAMEPAD_Y = 0x8000;
|
||||||
|
|
||||||
|
struct XINPUT_STATE {
|
||||||
|
DWORD dwPacketNumber;
|
||||||
|
XINPUT_GAMEPAD Gamepad;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct XINPUT_VIBRATION {
|
||||||
|
WORD wLeftMotorSpeed; // low frequency motor
|
||||||
|
WORD wRightMotorSpeed; // high frequency motor
|
||||||
|
}
|
||||||
|
|
||||||
|
struct WindowXInput {
|
||||||
|
HANDLE dll;
|
||||||
|
bool loadDll() {
|
||||||
|
// try Windows 8 first
|
||||||
|
dll = LoadLibraryA("Xinput1_4.dll");
|
||||||
|
if(dll is null) // then try Windows Vista
|
||||||
|
dll = LoadLibraryA("Xinput9_1_0.dll");
|
||||||
|
|
||||||
|
if(dll is null)
|
||||||
|
return false; // couldn't load it, tell user
|
||||||
|
|
||||||
|
XInputGetState = cast(typeof(XInputGetState)) GetProcAddress(dll, "XInputGetState");
|
||||||
|
XInputSetState = cast(typeof(XInputSetState)) GetProcAddress(dll, "XInputSetState");
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
~this() {
|
||||||
|
if(dll !is null)
|
||||||
|
FreeLibrary(dll);
|
||||||
|
}
|
||||||
|
|
||||||
|
// These are all dynamically loaded from the DLL
|
||||||
|
extern(Windows) {
|
||||||
|
DWORD function(DWORD, XINPUT_STATE*) XInputGetState;
|
||||||
|
DWORD function(DWORD, XINPUT_VIBRATION*) XInputSetState;
|
||||||
|
}
|
||||||
|
|
||||||
|
// there's other functions but I don't use them; my controllers
|
||||||
|
// are corded, for example, and I don't have a headset that works
|
||||||
|
// with them. But if I get ones, I'll add them too.
|
||||||
|
//
|
||||||
|
// There's also some Windows 8 and up functions I didn't use, I just
|
||||||
|
// wanted the basics.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
version(linux) {
|
||||||
|
|
||||||
|
// https://www.kernel.org/doc/Documentation/input/joystick-api.txt
|
||||||
|
struct js_event {
|
||||||
|
uint time;
|
||||||
|
short value;
|
||||||
|
ubyte type;
|
||||||
|
ubyte number;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum JS_EVENT_BUTTON = 0x01;
|
||||||
|
enum JS_EVENT_AXIS = 0x02;
|
||||||
|
enum JS_EVENT_INIT = 0x80;
|
||||||
|
|
||||||
|
import core.sys.posix.unistd;
|
||||||
|
import core.sys.posix.fcntl;
|
||||||
|
|
||||||
|
import std.stdio;
|
||||||
|
|
||||||
|
struct RawControllerEvent {
|
||||||
|
int controller;
|
||||||
|
int type;
|
||||||
|
int number;
|
||||||
|
int value;
|
||||||
|
}
|
||||||
|
|
||||||
|
// These values are determined experimentally on my Linux box
|
||||||
|
// and won't necessarily match what you have. I really don't know.
|
||||||
|
// TODO: see if these line up on Windows
|
||||||
|
|
||||||
|
// My hardware:
|
||||||
|
// a Sony PS1 dual shock controller on a PSX to USB adapter from Radio Shack
|
||||||
|
// and a wired XBox 360 controller from Microsoft.
|
||||||
|
|
||||||
|
enum PS1Buttons {
|
||||||
|
triangle = 0,
|
||||||
|
circle,
|
||||||
|
cross,
|
||||||
|
square,
|
||||||
|
l2,
|
||||||
|
r2,
|
||||||
|
l1,
|
||||||
|
r1,
|
||||||
|
select,
|
||||||
|
start,
|
||||||
|
l3,
|
||||||
|
r3
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use if analog is turned off
|
||||||
|
// Tip: if you just check this OR the analog one it will work in both cases easily enough
|
||||||
|
enum PS1Axes {
|
||||||
|
horizontalDpad = 0,
|
||||||
|
verticalDpad = 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use if analog is turned on
|
||||||
|
enum PS1AnalogAxes {
|
||||||
|
horiziontalLeftStick = 0,
|
||||||
|
verticalLeftStick,
|
||||||
|
verticalRightStick,
|
||||||
|
horiziontalRightStick,
|
||||||
|
horizontalDpad,
|
||||||
|
verticalDpad,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
enum XBox360Buttons {
|
||||||
|
a = 0,
|
||||||
|
b,
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
lb,
|
||||||
|
rb,
|
||||||
|
back,
|
||||||
|
start,
|
||||||
|
xboxLogo,
|
||||||
|
leftStick,
|
||||||
|
rightStick
|
||||||
|
}
|
||||||
|
|
||||||
|
enum XBox360Axes {
|
||||||
|
horizontalLeftStick = 0,
|
||||||
|
verticalLeftStick,
|
||||||
|
lt,
|
||||||
|
horizontalRightStick,
|
||||||
|
verticalLeftStick,
|
||||||
|
rt,
|
||||||
|
horizontalDpad,
|
||||||
|
verticalDpad
|
||||||
|
}
|
||||||
|
|
||||||
|
void main(string[] args) {
|
||||||
|
int fd = open(args.length > 1 ? (args[1]~'\0').ptr : "/dev/input/js0".ptr, O_RDONLY);
|
||||||
|
assert(fd > 0);
|
||||||
|
js_event event;
|
||||||
|
|
||||||
|
short[8] axes;
|
||||||
|
ubyte[16] buttons;
|
||||||
|
|
||||||
|
printf("\n");
|
||||||
|
|
||||||
|
while(true) {
|
||||||
|
int r = read(fd, &event, event.sizeof);
|
||||||
|
assert(r == event.sizeof);
|
||||||
|
|
||||||
|
// writef("\r%12s", event);
|
||||||
|
if(event.type & JS_EVENT_AXIS) {
|
||||||
|
axes[event.number] = event.value >> 12;
|
||||||
|
}
|
||||||
|
if(event.type & JS_EVENT_BUTTON) {
|
||||||
|
buttons[event.number] = event.value;
|
||||||
|
}
|
||||||
|
writef("\r%6s %1s", axes[0..8], buttons[0 .. 16]);
|
||||||
|
stdout.flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
close(fd);
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue