fixed broken driving controller support (see #760)

This commit is contained in:
thrust26 2021-02-02 15:44:28 +01:00
parent a7bde4603e
commit 6e6a1f4b43
8 changed files with 396 additions and 181 deletions

View File

@ -35,9 +35,10 @@ using json = nlohmann::json;
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
PhysicalJoystickHandler::PhysicalJoystickHandler( PhysicalJoystickHandler::PhysicalJoystickHandler(
OSystem& system, EventHandler& handler) OSystem& system, EventHandler& handler, Event& event)
: myOSystem{system}, : myOSystem{system},
myHandler{handler} myHandler{handler},
myEvent(event)
{ {
if(myOSystem.settings().getInt("event_ver") != Event::VERSION) { if(myOSystem.settings().getInt("event_ver") != Event::VERSION) {
Logger::info("event version mismatch; dropping previous joystick mappings"); Logger::info("event version mismatch; dropping previous joystick mappings");
@ -161,6 +162,15 @@ int PhysicalJoystickHandler::add(const PhysicalJoystickPtr& stick)
setStickDefaultMapping(stick->ID, Event::NoType, EventMode::kMenuMode); setStickDefaultMapping(stick->ID, Event::NoType, EventMode::kMenuMode);
} }
// We're potentially swapping out an input device behind the back of
// the Event system, so we make sure all Stelladaptor-generated events
// are reset
for(int port = 0; port < NUM_PORTS; ++port)
{
for(int axis = 0; axis < NUM_SA_AXIS; ++axis)
myEvent.set(SA_Axis[port][axis], 0);
}
return stick->ID; return stick->ID;
} }
@ -222,6 +232,7 @@ void PhysicalJoystickHandler::mapStelladaptors(const string& saport)
// in setupJoysticks take care of that // in setupJoysticks take care of that
int saCount = 0; int saCount = 0;
int saOrder[] = { 1, 2 }; int saOrder[] = { 1, 2 };
if(BSPF::equalsIgnoreCase(saport, "rl")) if(BSPF::equalsIgnoreCase(saport, "rl"))
{ {
saOrder[0] = 2; saOrder[1] = 1; saOrder[0] = 2; saOrder[1] = 1;
@ -229,6 +240,7 @@ void PhysicalJoystickHandler::mapStelladaptors(const string& saport)
for(auto& [_id, _joyptr]: mySticks) for(auto& [_id, _joyptr]: mySticks)
{ {
bool found = false;
// remove previously added emulated ports // remove previously added emulated ports
size_t pos = _joyptr->name.find(" (emulates "); size_t pos = _joyptr->name.find(" (emulates ");
@ -238,30 +250,29 @@ void PhysicalJoystickHandler::mapStelladaptors(const string& saport)
if(BSPF::startsWithIgnoreCase(_joyptr->name, "Stelladaptor")) if(BSPF::startsWithIgnoreCase(_joyptr->name, "Stelladaptor"))
{ {
if(saOrder[saCount] == 1) if(saOrder[saCount] == 1)
{
_joyptr->name += " (emulates left joystick port)";
_joyptr->type = PhysicalJoystick::Type::LEFT_STELLADAPTOR; _joyptr->type = PhysicalJoystick::Type::LEFT_STELLADAPTOR;
}
else if(saOrder[saCount] == 2) else if(saOrder[saCount] == 2)
{
_joyptr->name += " (emulates right joystick port)";
_joyptr->type = PhysicalJoystick::Type::RIGHT_STELLADAPTOR; _joyptr->type = PhysicalJoystick::Type::RIGHT_STELLADAPTOR;
} found = true;
saCount++;
} }
else if(BSPF::startsWithIgnoreCase(_joyptr->name, "2600-daptor")) else if(BSPF::startsWithIgnoreCase(_joyptr->name, "2600-daptor"))
{ {
if(saOrder[saCount] == 1) if(saOrder[saCount] == 1)
{
_joyptr->name += " (emulates left joystick port)";
_joyptr->type = PhysicalJoystick::Type::LEFT_2600DAPTOR; _joyptr->type = PhysicalJoystick::Type::LEFT_2600DAPTOR;
}
else if(saOrder[saCount] == 2) else if(saOrder[saCount] == 2)
{
_joyptr->name += " (emulates right joystick port)";
_joyptr->type = PhysicalJoystick::Type::RIGHT_2600DAPTOR; _joyptr->type = PhysicalJoystick::Type::RIGHT_2600DAPTOR;
} found = true;
}
if(found)
{
if(saOrder[saCount] == 1)
_joyptr->name += " (emulates left joystick port)";
else if(saOrder[saCount] == 2)
_joyptr->name += " (emulates right joystick port)";
saCount++; saCount++;
// always map Stelladaptor/2600-daptor to emulation mode defaults
setStickDefaultMapping(_joyptr->ID, Event::NoType, EventMode::kEmulationMode);
} }
} }
myOSystem.settings().setValue("saport", saport); myOSystem.settings().setValue("saport", saport);
@ -704,75 +715,185 @@ void PhysicalJoystickHandler::handleAxisEvent(int stick, int axis, int value)
if(j) if(j)
{ {
int button = j->buttonLast[stick]; //int button = j->buttonLast[stick];
if(myHandler.state() == EventHandlerState::EMULATION) switch(j->type)
{ {
Event::Type eventAxisAnalog; // Since the various controller classes deal with Stelladaptor
// devices differently, we send the raw X and Y axis data directly,
// and let the controller handle it
// These events don't have to pass through handleEvent, since
// they can never be remapped
case PhysicalJoystick::Type::LEFT_STELLADAPTOR:
case PhysicalJoystick::Type::LEFT_2600DAPTOR:
if(myOSystem.hasConsole()
&& myOSystem.console().leftController().type() == Controller::Type::Driving)
{
if(axis < NUM_SA_AXIS)
myEvent.set(SA_Axis[0][axis], value);
}
else
handleRegularAxisEvent(j, stick, axis, value);
break; // axis on left controller (0)
// Check for analog events, which are handled differently case PhysicalJoystick::Type::RIGHT_STELLADAPTOR:
// A value change lower than ~90% indicates analog input case PhysicalJoystick::Type::RIGHT_2600DAPTOR:
if((abs(j->axisLastValue[axis] - value) < 30000) if(myOSystem.hasConsole()
&& (eventAxisAnalog = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::ANALOG)) != Event::Type::NoType) && myOSystem.console().rightController().type() == Controller::Type::Driving)
{ {
myHandler.handleEvent(eventAxisAnalog, value); if(axis < NUM_SA_AXIS)
} myEvent.set(SA_Axis[1][axis], value);
}
else
handleRegularAxisEvent(j, stick, axis, value);
break; // axis on right controller (1)
default: // PhysicalJoystick::Type::REGULAR
handleRegularAxisEvent(j, stick, axis, value);
#if 0
// if(myHandler.state() == EventHandlerState::EMULATION)
// {
// Event::Type eventAxisAnalog;
// // Check for analog events, which are handled differently
// // A value change lower than ~90% indicates analog input
// if((abs(j->axisLastValue[axis] - value) < 30000)
// && (eventAxisAnalog = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::ANALOG)) != Event::Type::NoType)
// {
// myHandler.handleEvent(eventAxisAnalog, value);
// }
// else
// {
// // Otherwise, we assume the event is digital
// // Every axis event has two associated values, negative and positive
// Event::Type eventAxisNeg = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::NEG);
// Event::Type eventAxisPos = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::POS);
// if(value > Joystick::deadzone())
// myHandler.handleEvent(eventAxisPos);
// else if(value < -Joystick::deadzone())
// myHandler.handleEvent(eventAxisNeg);
// else
// {
// // Treat any deadzone value as zero
// value = 0;
// // Now filter out consecutive, similar values
// // (only pass on the event if the state has changed)
// if(j->axisLastValue[axis] != value)
// {
// // Turn off both events, since we don't know exactly which one
// // was previously activated.
// myHandler.handleEvent(eventAxisNeg, false);
// myHandler.handleEvent(eventAxisPos, false);
// }
// }
// }
// j->axisLastValue[axis] = value;
// }
// #ifdef GUI_SUPPORT
// else if(myHandler.hasOverlay())
// {
// // A value change lower than Joystick::deadzone indicates analog input which is ignored
// if((abs(j->axisLastValue[axis] - value) > Joystick::deadzone()))
// {
// // First, clamp the values to simulate digital input
// // (the only thing that the underlying code understands)
// if(value > Joystick::deadzone())
// value = 32000;
// else if(value < -Joystick::deadzone())
// value = -32000;
// else
// value = 0;
// // Now filter out consecutive, similar values
// // (only pass on the event if the state has changed)
// if(value != j->axisLastValue[axis])
// {
// myHandler.overlay().handleJoyAxisEvent(stick, JoyAxis(axis), convertAxisValue(value), button);
// }
// }
// j->axisLastValue[axis] = value;
// }
// #endif
#endif
}
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void PhysicalJoystickHandler::handleRegularAxisEvent(const PhysicalJoystickPtr j,
int stick, int axis, int value)
{
int button = j->buttonLast[stick];
if(myHandler.state() == EventHandlerState::EMULATION)
{
Event::Type eventAxisAnalog;
// Check for analog events, which are handled differently
// A value change lower than ~90% indicates analog input
if((abs(j->axisLastValue[axis] - value) < 30000)
&& (eventAxisAnalog = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::ANALOG)) != Event::Type::NoType)
{
myHandler.handleEvent(eventAxisAnalog, value);
}
else
{
// Otherwise, we assume the event is digital
// Every axis event has two associated values, negative and positive
Event::Type eventAxisNeg = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::NEG);
Event::Type eventAxisPos = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::POS);
if(value > Joystick::deadzone())
myHandler.handleEvent(eventAxisPos);
else if(value < -Joystick::deadzone())
myHandler.handleEvent(eventAxisNeg);
else else
{ {
// Otherwise, we assume the event is digital // Treat any deadzone value as zero
// Every axis event has two associated values, negative and positive value = 0;
Event::Type eventAxisNeg = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::NEG);
Event::Type eventAxisPos = j->joyMap.get(EventMode::kEmulationMode, button, JoyAxis(axis), JoyDir::POS);
if(value > Joystick::deadzone())
myHandler.handleEvent(eventAxisPos);
else if(value < -Joystick::deadzone())
myHandler.handleEvent(eventAxisNeg);
else
{
// Treat any deadzone value as zero
value = 0;
// Now filter out consecutive, similar values
// (only pass on the event if the state has changed)
if(j->axisLastValue[axis] != value)
{
// Turn off both events, since we don't know exactly which one
// was previously activated.
myHandler.handleEvent(eventAxisNeg, false);
myHandler.handleEvent(eventAxisPos, false);
}
}
}
j->axisLastValue[axis] = value;
}
#ifdef GUI_SUPPORT
else if(myHandler.hasOverlay())
{
// A value change lower than Joystick::deadzone indicates analog input which is ignored
if((abs(j->axisLastValue[axis] - value) > Joystick::deadzone()))
{
// First, clamp the values to simulate digital input
// (the only thing that the underlying code understands)
if(value > Joystick::deadzone())
value = 32000;
else if(value < -Joystick::deadzone())
value = -32000;
else
value = 0;
// Now filter out consecutive, similar values // Now filter out consecutive, similar values
// (only pass on the event if the state has changed) // (only pass on the event if the state has changed)
if(value != j->axisLastValue[axis]) if(j->axisLastValue[axis] != value)
{ {
myHandler.overlay().handleJoyAxisEvent(stick, JoyAxis(axis), convertAxisValue(value), button); // Turn off both events, since we don't know exactly which one
// was previously activated.
myHandler.handleEvent(eventAxisNeg, false);
myHandler.handleEvent(eventAxisPos, false);
} }
} }
j->axisLastValue[axis] = value;
} }
#endif j->axisLastValue[axis] = value;
} }
#ifdef GUI_SUPPORT
else if(myHandler.hasOverlay())
{
// A value change lower than Joystick::deadzone indicates analog input which is ignored
if((abs(j->axisLastValue[axis] - value) > Joystick::deadzone()))
{
// First, clamp the values to simulate digital input
// (the only thing that the underlying code understands)
if(value > Joystick::deadzone())
value = 32000;
else if(value < -Joystick::deadzone())
value = -32000;
else
value = 0;
// Now filter out consecutive, similar values
// (only pass on the event if the state has changed)
if(value != j->axisLastValue[axis])
{
myHandler.overlay().handleJoyAxisEvent(stick, JoyAxis(axis), convertAxisValue(value), button);
}
}
j->axisLastValue[axis] = value;
}
#endif
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -1160,3 +1281,10 @@ PhysicalJoystickHandler::DefaultMenuMapping = {
{Event::UIUp, JOY_CTRL_NONE, JoyAxis::NONE, JoyDir::NONE, 0, JoyHatDir::UP}, {Event::UIUp, JOY_CTRL_NONE, JoyAxis::NONE, JoyDir::NONE, 0, JoyHatDir::UP},
{Event::UIDown, JOY_CTRL_NONE, JoyAxis::NONE, JoyDir::NONE, 0, JoyHatDir::DOWN}, {Event::UIDown, JOY_CTRL_NONE, JoyAxis::NONE, JoyDir::NONE, 0, JoyHatDir::DOWN},
}; };
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// Used by the Stelladaptor to send absolute axis values
const Event::Type PhysicalJoystickHandler::SA_Axis[NUM_PORTS][NUM_SA_AXIS] = {
{ Event::SALeftAxis0Value, Event::SALeftAxis1Value },
{ Event::SARightAxis0Value, Event::SARightAxis1Value }
};

View File

@ -62,7 +62,7 @@ class PhysicalJoystickHandler
}; };
public: public:
PhysicalJoystickHandler(OSystem& system, EventHandler& handler); PhysicalJoystickHandler(OSystem& system, EventHandler& handler, Event& event);
static nlohmann::json convertLegacyMapping(const string& mapping); static nlohmann::json convertLegacyMapping(const string& mapping);
@ -125,6 +125,7 @@ class PhysicalJoystickHandler
OSystem& myOSystem; OSystem& myOSystem;
EventHandler& myHandler; EventHandler& myHandler;
Event& myEvent;
// Contains all joysticks that Stella knows about, indexed by name // Contains all joysticks that Stella knows about, indexed by name
StickDatabase myDatabase; StickDatabase myDatabase;
@ -149,6 +150,10 @@ class PhysicalJoystickHandler
return value == int(JoyDir::NONE) ? JoyDir::NONE : value > 0 ? JoyDir::POS : JoyDir::NEG; return value == int(JoyDir::NONE) ? JoyDir::NONE : value > 0 ? JoyDir::POS : JoyDir::NEG;
} }
// Handle regular axis events (besides special Stelladaptor handling)
void handleRegularAxisEvent(const PhysicalJoystickPtr j,
int stick, int axis, int value);
// Structures used for action menu items // Structures used for action menu items
struct EventMapping { struct EventMapping {
Event::Type event{Event::NoType}; Event::Type event{Event::NoType};
@ -192,6 +197,10 @@ class PhysicalJoystickHandler
static EventMappingArray DefaultRightPaddlesMapping; static EventMappingArray DefaultRightPaddlesMapping;
static EventMappingArray DefaultLeftKeypadMapping; static EventMappingArray DefaultLeftKeypadMapping;
static EventMappingArray DefaultRightKeypadMapping; static EventMappingArray DefaultRightKeypadMapping;
static constexpr int NUM_PORTS = 2;
static constexpr int NUM_SA_AXIS = 2;
static const Event::Type SA_Axis[NUM_PORTS][NUM_SA_AXIS];
}; };
#endif #endif

View File

@ -15,9 +15,6 @@
// this file, and for a DISCLAIMER OF ALL WARRANTIES. // this file, and for a DISCLAIMER OF ALL WARRANTIES.
//============================================================================ //============================================================================
#include "Event.hxx"
#include "System.hxx"
#include "Driving.hxx" #include "Driving.hxx"
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -38,8 +35,8 @@ Driving::Driving(Jack jack, const Event& event, const System& system, bool altma
myCWEvent = Event::JoystickTwoRight; myCWEvent = Event::JoystickTwoRight;
myFireEvent = Event::JoystickTwoFire; myFireEvent = Event::JoystickTwoFire;
} }
myXAxisValue = Event::PaddleZeroAnalog; myXAxisValue = Event::SALeftAxis0Value; // joystick input
myYAxisValue = Event::PaddleOneAnalog; myYAxisValue = Event::SALeftAxis1Value; // driving controller input
} }
else else
{ {
@ -55,8 +52,8 @@ Driving::Driving(Jack jack, const Event& event, const System& system, bool altma
myCWEvent = Event::JoystickThreeRight; myCWEvent = Event::JoystickThreeRight;
myFireEvent = Event::JoystickThreeFire; myFireEvent = Event::JoystickThreeFire;
} }
myXAxisValue = Event::PaddleTwoAnalog; myXAxisValue = Event::SARightAxis0Value; // joystick input
myYAxisValue = Event::PaddleThreeAnalog; myYAxisValue = Event::SARightAxis1Value; // driving controller input
} }
// Digital pins 3 and 4 are not connected // Digital pins 3 and 4 are not connected
@ -67,70 +64,11 @@ Driving::Driving(Jack jack, const Event& event, const System& system, bool altma
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Driving::update() void Driving::update()
{ {
// Digital events (from keyboard or joystick hats & buttons) updateButtons();
bool firePressed = myEvent.get(myFireEvent) != 0;
int d_axis = myEvent.get(myXAxisValue); updateDigitalAxes();
if(myEvent.get(myCCWEvent) != 0 || d_axis < -16384) --myCounter; updateMouseAxes();
else if(myEvent.get(myCWEvent) != 0 || d_axis > 16384) ++myCounter; updateStelladaptorAxes();
// Mouse motion and button events
if(myControlID > -1)
{
int m_axis = myEvent.get(Event::MouseAxisXMove);
if(m_axis < -2) --myCounter;
else if(m_axis > 2) ++myCounter;
firePressed = firePressed
|| myEvent.get(Event::MouseButtonLeftValue)
|| myEvent.get(Event::MouseButtonRightValue);
}
else
{
// Test for 'untied' mouse axis mode, where each axis is potentially
// mapped to a separate driving controller
if(myControlIDX > -1)
{
int m_axis = myEvent.get(Event::MouseAxisXMove);
if(m_axis < -2) --myCounter;
else if(m_axis > 2) ++myCounter;
firePressed = firePressed
|| myEvent.get(Event::MouseButtonLeftValue);
}
if(myControlIDY > -1)
{
int m_axis = myEvent.get(Event::MouseAxisYMove);
if(m_axis < -2) --myCounter;
else if(m_axis > 2) ++myCounter;
firePressed = firePressed
|| myEvent.get(Event::MouseButtonRightValue);
}
}
setPin(DigitalPin::Six, !getAutoFireState(firePressed));
// Only consider the lower-most bits (corresponding to pins 1 & 2)
myGrayIndex = Int32(myCounter * SENSITIVITY / 4.0F) & 0b11;
// Stelladaptor is the only controller that should set this
int yaxis = myEvent.get(myYAxisValue);
// Only overwrite gray code when Stelladaptor input has changed
// (that means real changes, not just analog signal jitter)
if((yaxis < (myLastYaxis - 1024)) || (yaxis > (myLastYaxis + 1024)))
{
myLastYaxis = yaxis;
if(yaxis <= -16384-4096)
myGrayIndex = 3; // up
else if(yaxis > 16384+4096)
myGrayIndex = 1; // down
else if(yaxis >= 16384-4096)
myGrayIndex = 2; // up + down
else /* if(yaxis < 16384-4096) */
myGrayIndex = 0; // no movement
// Make sure direct gray codes from Stelladaptor stay in sync with
// simulated gray codes generated by PC keyboard or PC joystick
myCounter = myGrayIndex / SENSITIVITY * 4.0F;
}
// Gray codes for rotation // Gray codes for rotation
static constexpr std::array<uInt8, 4> graytable = { 0x03, 0x01, 0x00, 0x02 }; static constexpr std::array<uInt8, 4> graytable = { 0x03, 0x01, 0x00, 0x02 };
@ -141,6 +79,113 @@ void Driving::update()
setPin(DigitalPin::Two, (gray & 0x2) != 0); setPin(DigitalPin::Two, (gray & 0x2) != 0);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Driving::updateButtons()
{
bool firePressed = myEvent.get(myFireEvent) != 0;
// The joystick uses both mouse buttons for the single joystick button
updateMouseButtons(firePressed);
setPin(DigitalPin::Six, !getAutoFireState(firePressed));
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Driving::updateMouseButtons(bool& firePressed)
{
if(myControlID > -1)
firePressed |= (myEvent.get(Event::MouseButtonLeftValue) != 0
|| myEvent.get(Event::MouseButtonRightValue) != 0);
else
{
// Test for 'untied' mouse axis mode, where each axis is potentially
// mapped to a separate driving controller
if(myControlIDX > -1)
firePressed |= (myEvent.get(Event::MouseButtonLeftValue) != 0);
if(myControlIDY > -1)
firePressed |= (myEvent.get(Event::MouseButtonRightValue) != 0);
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Driving::updateDigitalAxes()
{
// Digital events (from keyboard or joystick hats & buttons)
int d_axis = myEvent.get(myXAxisValue);
if(myEvent.get(myCCWEvent) != 0 || d_axis < -16384)
--myCounter;
else if(myEvent.get(myCWEvent) != 0 || d_axis > 16384)
++myCounter;
// Only consider the lower-most bits (corresponding to pins 1 & 2)
myGrayIndex = Int32(myCounter * SENSITIVITY / 4.0F) & 0b11;
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Driving::updateMouseAxes()
{
#define MJ_Threshold 2
// Mouse motion and button events
if(myControlID > -1)
{
int m_axis = myEvent.get(Event::MouseAxisXMove);
if(m_axis < -MJ_Threshold)
--myCounter;
else if(m_axis > MJ_Threshold)
++myCounter;
}
else
{
// Test for 'untied' mouse axis mode, where each axis is potentially
// mapped to a separate driving controller
if(myControlIDX > -1)
{
int m_axis = myEvent.get(Event::MouseAxisXMove);
if(m_axis < -MJ_Threshold)
--myCounter;
else if(m_axis > MJ_Threshold)
++myCounter;
}
if(myControlIDY > -1)
{
int m_axis = myEvent.get(Event::MouseAxisYMove);
if(m_axis < -MJ_Threshold)
--myCounter;
else if(m_axis > MJ_Threshold)
++myCounter;
}
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Driving::updateStelladaptorAxes()
{
#define SA_Threshold 4096
// Stelladaptor is the only controller that should set this
int yaxis = myEvent.get(myYAxisValue);
// Only overwrite gray code when Stelladaptor input has changed
// (that means real changes, not just analog signal jitter)
if((yaxis < (myLastYaxis - 1024)) || (yaxis > (myLastYaxis + 1024)))
{
myLastYaxis = yaxis;
if(yaxis <= -16384 - SA_Threshold)
myGrayIndex = 3; // up
else if(yaxis > 16384 + SA_Threshold)
myGrayIndex = 1; // down
else if(yaxis >= 16384 - SA_Threshold)
myGrayIndex = 2; // up + down
else /* if(yaxis < 16384 - SA_Threshold) */
myGrayIndex = 0; // no movement
// Make sure direct gray codes from Stelladaptor stay in sync with
// simulated gray codes generated by PC keyboard or PC joystick
// Must be rounded into the middle of the myCounter interval!
myCounter = (myGrayIndex + 0.5F) * 4.0F / SENSITIVITY;
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
bool Driving::setMouseControl( bool Driving::setMouseControl(
Controller::Type xtype, int xid, Controller::Type ytype, int yid) Controller::Type xtype, int xid, Controller::Type ytype, int yid)

View File

@ -116,6 +116,35 @@ class Driving : public Controller
// speeds // speeds
static float SENSITIVITY; static float SENSITIVITY;
private:
/**
Update the button pin states.
*/
void updateButtons();
/**
Update the button states from the mouse button events currently set.
*/
void updateMouseButtons(bool& firePressed);
/**
Update the axes pin states according to the keyboard
or joystick hats & buttons events currently set.
*/
void updateDigitalAxes();
/**
Update the axes pin states according to the Stelladaptor axes value
events currently set.
*/
void updateStelladaptorAxes();
/**
Update the axes pin states according to mouse events currently set.
*/
void updateMouseAxes();
private: private:
// Following constructors and assignment operators not supported // Following constructors and assignment operators not supported
Driving() = delete; Driving() = delete;

View File

@ -165,6 +165,7 @@ class Event
DecreasePaddleCenterY, IncreasePaddleCenterY, DecreasePaddleCenterY, IncreasePaddleCenterY,
PreviousMouseControl, PreviousMouseControl,
DecreaseMouseAxesRange, IncreaseMouseAxesRange, DecreaseMouseAxesRange, IncreaseMouseAxesRange,
SALeftAxis0Value, SALeftAxis1Value, SARightAxis0Value, SARightAxis1Value,
LastType LastType
}; };

View File

@ -87,7 +87,7 @@ void EventHandler::initialize()
myPKeyHandler = make_unique<PhysicalKeyboardHandler>(myOSystem, *this); myPKeyHandler = make_unique<PhysicalKeyboardHandler>(myOSystem, *this);
// Create joystick handler (to handle all physical joystick functionality) // Create joystick handler (to handle all physical joystick functionality)
myPJoyHandler = make_unique<PhysicalJoystickHandler>(myOSystem, *this); myPJoyHandler = make_unique<PhysicalJoystickHandler>(myOSystem, *this, myEvent);
// Erase the 'combo' array // Erase the 'combo' array
for(int i = 0; i < COMBO_SIZE; ++i) for(int i = 0; i < COMBO_SIZE; ++i)

View File

@ -46,8 +46,8 @@ Joystick::Joystick(Jack jack, const Event& event, const System& system,
myRightEvent = Event::JoystickTwoRight; myRightEvent = Event::JoystickTwoRight;
myFireEvent = Event::JoystickTwoFire; myFireEvent = Event::JoystickTwoFire;
} }
myXAxisValue = Event::PaddleZeroAnalog; // TODO //myXAxisValue = Event::SALeftAxis0Value;
myYAxisValue = Event::PaddleOneAnalog; // TODO //myYAxisValue = Event::SALeftAxis1Value;
} }
else else
{ {
@ -67,8 +67,8 @@ Joystick::Joystick(Jack jack, const Event& event, const System& system,
myRightEvent = Event::JoystickThreeRight; myRightEvent = Event::JoystickThreeRight;
myFireEvent = Event::JoystickThreeFire; myFireEvent = Event::JoystickThreeFire;
} }
myXAxisValue = Event::PaddleTwoAnalog; // TODO //myXAxisValue = Event::SARightAxis0Value;
myYAxisValue = Event::PaddleThreeAnalog; // TODO //myYAxisValue = Event::SARightAxis1Value;
} }
} }
@ -78,7 +78,7 @@ void Joystick::update()
updateButtons(); updateButtons();
updateDigitalAxes(); updateDigitalAxes();
updateAnalogAxes(); //updateStelladaptorAxes();
updateMouseAxes(); updateMouseAxes();
} }
@ -110,34 +110,36 @@ void Joystick::updateDigitalAxes()
setPin(DigitalPin::One, myEvent.get(myUpEvent) == 0); setPin(DigitalPin::One, myEvent.get(myUpEvent) == 0);
setPin(DigitalPin::Two, myEvent.get(myDownEvent) == 0); setPin(DigitalPin::Two, myEvent.get(myDownEvent) == 0);
setPin(DigitalPin::Three, myEvent.get(myLeftEvent) == 0); setPin(DigitalPin::Three, myEvent.get(myLeftEvent) == 0);
setPin(DigitalPin::Four, myEvent.get(myRightEvent) == 0); setPin(DigitalPin::Four, myEvent.get(myRightEvent) != 1);
cerr << myEvent.get(myRightEvent) << endl;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - //// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Joystick::updateAnalogAxes() //void Joystick::updateStelladaptorAxes()
{ //{
// Axis events (usually generated by the Stelladaptor) // #define SA_Threshold 4096
int xaxis = myEvent.get(myXAxisValue); // // Axis events (usually generated by the Stelladaptor)
int yaxis = myEvent.get(myYAxisValue); // int xaxis = myEvent.get(myXAxisValue);
if(xaxis > 16384 - 4096) // int yaxis = myEvent.get(myYAxisValue);
{ // if(xaxis > 16384 - SA_Threshold)
setPin(DigitalPin::Four, false); // {
// Stelladaptor sends "half moved right" for L+R pushed together // setPin(DigitalPin::Four, false);
if(xaxis < 16384 + 4096) // // Stelladaptor sends "half moved right" for L+R pushed together
setPin(DigitalPin::Three, false); // if(xaxis < 16384 + SA_Threshold)
} // setPin(DigitalPin::Three, false);
else if(xaxis < -16384) // }
setPin(DigitalPin::Three, false); // else if(xaxis < -16384)
if(yaxis > 16384 - 4096) // setPin(DigitalPin::Three, false);
{ // if(yaxis > 16384 - SA_Threshold)
setPin(DigitalPin::Two, false); // {
// Stelladaptor sends "half moved down" for U+D pushed together // setPin(DigitalPin::Two, false);
if(yaxis < 16384 + 4096) // // Stelladaptor sends "half moved down" for U+D pushed together
setPin(DigitalPin::One, false); // if(yaxis < 16384 + SA_Threshold)
} // setPin(DigitalPin::One, false);
else if(yaxis < -16384) // }
setPin(DigitalPin::One, false); // else if(yaxis < -16384)
} // setPin(DigitalPin::One, false);
//}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Joystick::updateMouseAxes() void Joystick::updateMouseAxes()

View File

@ -18,8 +18,8 @@
#ifndef JOYSTICK_HXX #ifndef JOYSTICK_HXX
#define JOYSTICK_HXX #define JOYSTICK_HXX
#include "Event.hxx"
#include "Control.hxx" #include "Control.hxx"
#include "Event.hxx"
/** /**
The standard Atari 2600 joystick controller. The standard Atari 2600 joystick controller.
@ -118,8 +118,8 @@ class Joystick : public Controller
private: private:
// Pre-compute the events we care about based on given port // Pre-compute the events we care about based on given port
// This will eliminate test for left or right port in update() // This will eliminate test for left or right port in update()
Event::Type myUpEvent, myDownEvent, myLeftEvent, myRightEvent, Event::Type myUpEvent, myDownEvent, myLeftEvent, myRightEvent;
myXAxisValue, myYAxisValue; //myXAxisValue, myYAxisValue;
// Controller to emulate in normal mouse axis mode // Controller to emulate in normal mouse axis mode
int myControlID{-1}; int myControlID{-1};
@ -133,10 +133,11 @@ class Joystick : public Controller
*/ */
void updateDigitalAxes(); void updateDigitalAxes();
/** ///**
Update the axes pin states according to the axes value events currently set. // Update the axes pin states according to the Stelladaptor axes value
*/ // events currently set.
void updateAnalogAxes(); //*/
//void updateStelladaptorAxes();
/** /**
Update the axes pin states according to mouse events currently set. Update the axes pin states according to mouse events currently set.