2009-08-07 18:56:48 +00:00
|
|
|
// FGJoystickInput.cxx -- handle user input from joystick devices
|
|
|
|
//
|
|
|
|
// Written by Torsten Dreyer, started August 2009
|
|
|
|
// Based on work from David Megginson, started May 2001.
|
|
|
|
//
|
|
|
|
// Copyright (C) 2009 Torsten Dreyer, Torsten (at) t3r _dot_ de
|
|
|
|
// Copyright (C) 2001 David Megginson, david@megginson.com
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License as
|
|
|
|
// published by the Free Software Foundation; either version 2 of the
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful, but
|
|
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
//
|
|
|
|
// $Id$
|
|
|
|
|
|
|
|
#include "FGJoystickInput.hxx"
|
|
|
|
#include "FGDeviceConfigurationMap.hxx"
|
|
|
|
#include <Main/fg_props.hxx>
|
|
|
|
#include <Scripting/NasalSys.hxx>
|
|
|
|
|
2010-04-15 12:31:12 +00:00
|
|
|
using simgear::PropertyList;
|
|
|
|
|
2009-08-07 18:56:48 +00:00
|
|
|
FGJoystickInput::axis::axis ()
|
|
|
|
: last_value(9999999),
|
|
|
|
tolerance(0.002),
|
|
|
|
low_threshold(-0.9),
|
|
|
|
high_threshold(0.9),
|
|
|
|
interval_sec(0),
|
|
|
|
last_dt(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
FGJoystickInput::axis::~axis ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
FGJoystickInput::joystick::joystick ()
|
|
|
|
: jsnum(0),
|
|
|
|
js(0),
|
|
|
|
naxes(0),
|
|
|
|
nbuttons(0),
|
|
|
|
axes(0),
|
|
|
|
buttons(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
FGJoystickInput::joystick::~joystick ()
|
|
|
|
{
|
|
|
|
// delete js? why not?
|
|
|
|
// delete js;
|
|
|
|
delete[] axes;
|
|
|
|
delete[] buttons;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FGJoystickInput::FGJoystickInput()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
FGJoystickInput::~FGJoystickInput()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FGJoystickInput::init()
|
|
|
|
{
|
|
|
|
jsInit();
|
|
|
|
// TODO: zero the old bindings first.
|
|
|
|
SG_LOG(SG_INPUT, SG_DEBUG, "Initializing joystick bindings");
|
|
|
|
SGPropertyNode * js_nodes = fgGetNode("/input/joysticks", true);
|
|
|
|
|
|
|
|
FGDeviceConfigurationMap configMap("Input/Joysticks", js_nodes, "js-named");
|
|
|
|
|
|
|
|
for (int i = 0; i < MAX_JOYSTICKS; i++) {
|
|
|
|
jsJoystick * js = new jsJoystick(i);
|
|
|
|
bindings[i].js = js;
|
|
|
|
|
|
|
|
if (js->notWorking()) {
|
|
|
|
SG_LOG(SG_INPUT, SG_DEBUG, "Joystick " << i << " not found");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2011-01-14 22:37:39 +01:00
|
|
|
const char * name = js->getName();
|
2009-08-07 18:56:48 +00:00
|
|
|
SGPropertyNode_ptr js_node = js_nodes->getChild("js", i);
|
|
|
|
|
|
|
|
if (js_node) {
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Using existing bindings for joystick " << i);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Looking for bindings for joystick \"" << name << '"');
|
|
|
|
SGPropertyNode_ptr named;
|
|
|
|
|
|
|
|
if ((named = configMap[name])) {
|
|
|
|
string source = named->getStringValue("source", "user defined");
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "... found joystick: " << source);
|
|
|
|
|
|
|
|
} else if ((named = configMap["default"])) {
|
|
|
|
string source = named->getStringValue("source", "user defined");
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "No config found for joystick \"" << name
|
|
|
|
<< "\"\nUsing default: \"" << source << '"');
|
|
|
|
|
|
|
|
} else {
|
|
|
|
SG_LOG(SG_INPUT, SG_WARN, "No joystick configuration file with <name>" << name << "</name> entry found!");
|
|
|
|
}
|
|
|
|
|
|
|
|
js_node = js_nodes->getChild("js", i, true);
|
|
|
|
copyProperties(named, js_node);
|
|
|
|
js_node->setStringValue("id", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FGJoystickInput::postinit()
|
|
|
|
{
|
|
|
|
FGNasalSys *nasalsys = (FGNasalSys *)globals->get_subsystem("nasal");
|
|
|
|
SGPropertyNode_ptr js_nodes = fgGetNode("/input/joysticks");
|
|
|
|
|
|
|
|
for (int i = 0; i < MAX_JOYSTICKS; i++) {
|
|
|
|
SGPropertyNode_ptr js_node = js_nodes->getChild("js", i);
|
|
|
|
jsJoystick *js = bindings[i].js;
|
|
|
|
if (!js_node || js->notWorking())
|
|
|
|
continue;
|
|
|
|
|
2011-01-14 22:37:39 +01:00
|
|
|
#ifdef WIN32
|
|
|
|
JOYCAPS jsCaps ;
|
|
|
|
joyGetDevCaps( i, &jsCaps, sizeof(jsCaps) );
|
|
|
|
unsigned int nbuttons = jsCaps.wNumButtons;
|
2009-08-07 18:56:48 +00:00
|
|
|
if (nbuttons > MAX_JOYSTICK_BUTTONS) nbuttons = MAX_JOYSTICK_BUTTONS;
|
2011-01-14 22:37:39 +01:00
|
|
|
#else
|
|
|
|
unsigned int nbuttons = MAX_JOYSTICK_BUTTONS;
|
|
|
|
#endif
|
2009-08-07 18:56:48 +00:00
|
|
|
|
|
|
|
int naxes = js->getNumAxes();
|
|
|
|
if (naxes > MAX_JOYSTICK_AXES) naxes = MAX_JOYSTICK_AXES;
|
|
|
|
bindings[i].naxes = naxes;
|
|
|
|
bindings[i].nbuttons = nbuttons;
|
|
|
|
|
|
|
|
SG_LOG(SG_INPUT, SG_DEBUG, "Initializing joystick " << i);
|
|
|
|
|
|
|
|
// Set up range arrays
|
|
|
|
float minRange[MAX_JOYSTICK_AXES];
|
|
|
|
float maxRange[MAX_JOYSTICK_AXES];
|
|
|
|
float center[MAX_JOYSTICK_AXES];
|
|
|
|
|
|
|
|
// Initialize with default values
|
|
|
|
js->getMinRange(minRange);
|
|
|
|
js->getMaxRange(maxRange);
|
|
|
|
js->getCenter(center);
|
|
|
|
|
|
|
|
// Allocate axes and buttons
|
|
|
|
bindings[i].axes = new axis[naxes];
|
|
|
|
bindings[i].buttons = new FGButton[nbuttons];
|
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize nasal groups.
|
|
|
|
//
|
|
|
|
ostringstream str;
|
|
|
|
str << "__js" << i;
|
|
|
|
string module = str.str();
|
|
|
|
nasalsys->createModule(module.c_str(), module.c_str(), "", 0);
|
|
|
|
|
2010-04-15 12:31:12 +00:00
|
|
|
PropertyList nasal = js_node->getChildren("nasal");
|
2009-08-07 18:56:48 +00:00
|
|
|
unsigned int j;
|
|
|
|
for (j = 0; j < nasal.size(); j++) {
|
|
|
|
nasal[j]->setStringValue("module", module.c_str());
|
|
|
|
nasalsys->handleCommand(nasal[j]);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize the axes.
|
|
|
|
//
|
2010-04-15 12:31:12 +00:00
|
|
|
PropertyList axes = js_node->getChildren("axis");
|
2009-08-07 18:56:48 +00:00
|
|
|
size_t nb_axes = axes.size();
|
|
|
|
for (j = 0; j < nb_axes; j++ ) {
|
|
|
|
const SGPropertyNode * axis_node = axes[j];
|
|
|
|
const SGPropertyNode * num_node = axis_node->getChild("number");
|
|
|
|
int n_axis = axis_node->getIndex();
|
|
|
|
if (num_node != 0) {
|
|
|
|
n_axis = num_node->getIntValue(TGT_PLATFORM, -1);
|
|
|
|
|
|
|
|
// Silently ignore platforms that are not specified within the
|
|
|
|
// <number></number> section
|
|
|
|
if (n_axis < 0)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (n_axis >= naxes) {
|
|
|
|
SG_LOG(SG_INPUT, SG_DEBUG, "Dropping bindings for axis " << n_axis);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
axis &a = bindings[i].axes[n_axis];
|
|
|
|
|
|
|
|
js->setDeadBand(n_axis, axis_node->getDoubleValue("dead-band", 0.0));
|
|
|
|
|
|
|
|
a.tolerance = axis_node->getDoubleValue("tolerance", 0.002);
|
|
|
|
minRange[n_axis] = axis_node->getDoubleValue("min-range", minRange[n_axis]);
|
|
|
|
maxRange[n_axis] = axis_node->getDoubleValue("max-range", maxRange[n_axis]);
|
|
|
|
center[n_axis] = axis_node->getDoubleValue("center", center[n_axis]);
|
|
|
|
|
|
|
|
read_bindings(axis_node, a.bindings, KEYMOD_NONE, module );
|
|
|
|
|
|
|
|
// Initialize the virtual axis buttons.
|
|
|
|
a.low.init(axis_node->getChild("low"), "low", module );
|
|
|
|
a.low_threshold = axis_node->getDoubleValue("low-threshold", -0.9);
|
|
|
|
|
|
|
|
a.high.init(axis_node->getChild("high"), "high", module );
|
|
|
|
a.high_threshold = axis_node->getDoubleValue("high-threshold", 0.9);
|
|
|
|
a.interval_sec = axis_node->getDoubleValue("interval-sec",0.0);
|
|
|
|
a.last_dt = 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize the buttons.
|
|
|
|
//
|
2010-04-15 12:31:12 +00:00
|
|
|
PropertyList buttons = js_node->getChildren("button");
|
2009-08-07 18:56:48 +00:00
|
|
|
char buf[32];
|
|
|
|
for (j = 0; j < buttons.size() && j < nbuttons; j++) {
|
|
|
|
const SGPropertyNode * button_node = buttons[j];
|
|
|
|
const SGPropertyNode * num_node = button_node->getChild("number");
|
|
|
|
size_t n_but = button_node->getIndex();
|
|
|
|
if (num_node != 0) {
|
|
|
|
n_but = num_node->getIntValue(TGT_PLATFORM,n_but);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (n_but >= nbuttons) {
|
|
|
|
SG_LOG(SG_INPUT, SG_DEBUG, "Dropping bindings for button " << n_but);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-08-20 22:06:38 +00:00
|
|
|
sprintf(buf, "%u", (unsigned)n_but);
|
2009-08-07 18:56:48 +00:00
|
|
|
SG_LOG(SG_INPUT, SG_DEBUG, "Initializing button " << n_but);
|
|
|
|
bindings[i].buttons[n_but].init(button_node, buf, module );
|
|
|
|
|
|
|
|
// get interval-sec property
|
|
|
|
FGButton &b = bindings[i].buttons[n_but];
|
|
|
|
if (button_node != 0) {
|
|
|
|
b.interval_sec = button_node->getDoubleValue("interval-sec",0.0);
|
|
|
|
b.last_dt = 0.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
js->setMinRange(minRange);
|
|
|
|
js->setMaxRange(maxRange);
|
|
|
|
js->setCenter(center);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FGJoystickInput::update( double dt )
|
|
|
|
{
|
|
|
|
float axis_values[MAX_JOYSTICK_AXES];
|
|
|
|
int modifiers = fgGetKeyModifiers();
|
|
|
|
int buttons;
|
|
|
|
|
|
|
|
for (int i = 0; i < MAX_JOYSTICKS; i++) {
|
|
|
|
|
|
|
|
jsJoystick * js = bindings[i].js;
|
|
|
|
if (js == 0 || js->notWorking())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
js->read(&buttons, axis_values);
|
|
|
|
|
|
|
|
// Fire bindings for the axes.
|
|
|
|
for (int j = 0; j < bindings[i].naxes; j++) {
|
|
|
|
axis &a = bindings[i].axes[j];
|
|
|
|
|
|
|
|
// Do nothing if the axis position
|
|
|
|
// is unchanged; only a change in
|
|
|
|
// position fires the bindings.
|
|
|
|
if (fabs(axis_values[j] - a.last_value) > a.tolerance) {
|
|
|
|
a.last_value = axis_values[j];
|
|
|
|
for (unsigned int k = 0; k < a.bindings[KEYMOD_NONE].size(); k++)
|
|
|
|
a.bindings[KEYMOD_NONE][k]->fire(axis_values[j]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// do we have to emulate axis buttons?
|
|
|
|
a.last_dt += dt;
|
|
|
|
if(a.last_dt >= a.interval_sec) {
|
|
|
|
if (a.low.bindings[modifiers].size())
|
|
|
|
bindings[i].axes[j].low.update( modifiers, axis_values[j] < a.low_threshold );
|
|
|
|
|
|
|
|
if (a.high.bindings[modifiers].size())
|
|
|
|
bindings[i].axes[j].high.update( modifiers, axis_values[j] > a.high_threshold );
|
|
|
|
|
|
|
|
a.last_dt -= a.interval_sec;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fire bindings for the buttons.
|
|
|
|
for (int j = 0; j < bindings[i].nbuttons; j++) {
|
|
|
|
FGButton &b = bindings[i].buttons[j];
|
|
|
|
b.last_dt += dt;
|
|
|
|
if(b.last_dt >= b.interval_sec) {
|
|
|
|
bindings[i].buttons[j].update( modifiers, (buttons & (1 << j)) > 0 );
|
|
|
|
b.last_dt -= b.interval_sec;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|