mirror of https://github.com/stella-emu/stella.git
refactored Paddles class
This commit is contained in:
parent
d9c8750d01
commit
fb73d90cf9
|
@ -47,113 +47,50 @@ Paddles::Paddles(Jack jack, const Event& event, const System& system,
|
||||||
// Consider whether this is the left or right port
|
// Consider whether this is the left or right port
|
||||||
if(myJack == Jack::Left)
|
if(myJack == Jack::Left)
|
||||||
{
|
{
|
||||||
if(!swappaddle) // First paddle is 0, second is 1
|
// First paddle is 0, second is 1
|
||||||
{
|
myP0AxisValue = Event::PaddleZeroAnalog;
|
||||||
// These aren't affected by changes in axis orientation
|
myP1AxisValue = Event::PaddleOneAnalog;
|
||||||
myP0AxisValue = Event::PaddleZeroAnalog;
|
myP0FireEvent = Event::PaddleZeroFire;
|
||||||
myP1AxisValue = Event::PaddleOneAnalog;
|
myP1FireEvent = Event::PaddleOneFire;
|
||||||
myP0FireEvent = Event::PaddleZeroFire;
|
|
||||||
myP1FireEvent = Event::PaddleOneFire;
|
|
||||||
|
|
||||||
// Direction of movement is swapped
|
// These can be affected by changes in axis orientation
|
||||||
// That is, moving in a certain direction on an axis can
|
myP0DecEvent = Event::PaddleZeroDecrease;
|
||||||
// result in either increasing or decreasing paddle movement
|
myP0IncEvent = Event::PaddleZeroIncrease;
|
||||||
if(!swapdir)
|
myP1DecEvent = Event::PaddleOneDecrease;
|
||||||
{
|
myP1IncEvent = Event::PaddleOneIncrease;
|
||||||
myP0DecEvent = Event::PaddleZeroDecrease;
|
|
||||||
myP0IncEvent = Event::PaddleZeroIncrease;
|
|
||||||
myP1DecEvent = Event::PaddleOneDecrease;
|
|
||||||
myP1IncEvent = Event::PaddleOneIncrease;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
myP0DecEvent = Event::PaddleZeroIncrease;
|
|
||||||
myP0IncEvent = Event::PaddleZeroDecrease;
|
|
||||||
myP1DecEvent = Event::PaddleOneIncrease;
|
|
||||||
myP1IncEvent = Event::PaddleOneDecrease;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else // First paddle is 1, second is 0
|
|
||||||
{
|
|
||||||
// These aren't affected by changes in axis orientation
|
|
||||||
myP0AxisValue = Event::PaddleOneAnalog;
|
|
||||||
myP1AxisValue = Event::PaddleZeroAnalog;
|
|
||||||
myP0FireEvent = Event::PaddleOneFire;
|
|
||||||
myP1FireEvent = Event::PaddleZeroFire;
|
|
||||||
|
|
||||||
// Direction of movement is swapped
|
|
||||||
// That is, moving in a certain direction on an axis can
|
|
||||||
// result in either increasing or decreasing paddle movement
|
|
||||||
if(!swapdir)
|
|
||||||
{
|
|
||||||
myP0DecEvent = Event::PaddleOneDecrease;
|
|
||||||
myP0IncEvent = Event::PaddleOneIncrease;
|
|
||||||
myP1DecEvent = Event::PaddleZeroDecrease;
|
|
||||||
myP1IncEvent = Event::PaddleZeroIncrease;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
myP0DecEvent = Event::PaddleOneIncrease;
|
|
||||||
myP0IncEvent = Event::PaddleOneDecrease;
|
|
||||||
myP1DecEvent = Event::PaddleZeroIncrease;
|
|
||||||
myP1IncEvent = Event::PaddleZeroDecrease;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else // Jack is right port
|
else // Jack is right port
|
||||||
{
|
{
|
||||||
if(!swappaddle) // First paddle is 2, second is 3
|
// First paddle is 2, second is 3
|
||||||
{
|
myP0AxisValue = Event::PaddleTwoAnalog;
|
||||||
// These aren't affected by changes in axis orientation
|
myP1AxisValue = Event::PaddleThreeAnalog;
|
||||||
myP0AxisValue = Event::PaddleTwoAnalog;
|
myP0FireEvent = Event::PaddleTwoFire;
|
||||||
myP1AxisValue = Event::PaddleThreeAnalog;
|
myP1FireEvent = Event::PaddleThreeFire;
|
||||||
myP0FireEvent = Event::PaddleTwoFire;
|
|
||||||
myP1FireEvent = Event::PaddleThreeFire;
|
|
||||||
|
|
||||||
// Direction of movement is swapped
|
// These can be affected by changes in axis orientation
|
||||||
// That is, moving in a certain direction on an axis can
|
myP0DecEvent = Event::PaddleTwoDecrease;
|
||||||
// result in either increasing or decreasing paddle movement
|
myP0IncEvent = Event::PaddleTwoIncrease;
|
||||||
if(!swapdir)
|
myP1DecEvent = Event::PaddleThreeDecrease;
|
||||||
{
|
myP1IncEvent = Event::PaddleThreeIncrease;
|
||||||
myP0DecEvent = Event::PaddleTwoDecrease;
|
}
|
||||||
myP0IncEvent = Event::PaddleTwoIncrease;
|
|
||||||
myP1DecEvent = Event::PaddleThreeDecrease;
|
|
||||||
myP1IncEvent = Event::PaddleThreeIncrease;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
myP0DecEvent = Event::PaddleTwoIncrease;
|
|
||||||
myP0IncEvent = Event::PaddleTwoDecrease;
|
|
||||||
myP1DecEvent = Event::PaddleThreeIncrease;
|
|
||||||
myP1IncEvent = Event::PaddleThreeDecrease;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else // First paddle is 3, second is 2
|
|
||||||
{
|
|
||||||
// These aren't affected by changes in axis orientation
|
|
||||||
myP0AxisValue = Event::PaddleThreeAnalog;
|
|
||||||
myP1AxisValue = Event::PaddleTwoAnalog;
|
|
||||||
myP0FireEvent = Event::PaddleThreeFire;
|
|
||||||
myP1FireEvent = Event::PaddleTwoFire;
|
|
||||||
|
|
||||||
// Direction of movement is swapped
|
// Some games swap the paddles
|
||||||
// That is, moving in a certain direction on an axis can
|
if(swappaddle)
|
||||||
// result in either increasing or decreasing paddle movement
|
{
|
||||||
if(!swapdir)
|
// First paddle is 1|3, second is 0|2
|
||||||
{
|
swapEvents(myP0AxisValue, myP1AxisValue);
|
||||||
myP0DecEvent = Event::PaddleThreeDecrease;
|
swapEvents(myP0FireEvent, myP1FireEvent);
|
||||||
myP0IncEvent = Event::PaddleThreeIncrease;
|
swapEvents(myP0DecEvent, myP1DecEvent);
|
||||||
myP1DecEvent = Event::PaddleTwoDecrease;
|
swapEvents(myP0IncEvent, myP1IncEvent);
|
||||||
myP1IncEvent = Event::PaddleTwoIncrease;
|
}
|
||||||
}
|
|
||||||
else
|
// Direction of movement can be swapped
|
||||||
{
|
// That is, moving in a certain direction on an axis can
|
||||||
myP0DecEvent = Event::PaddleThreeIncrease;
|
// result in either increasing or decreasing paddle movement
|
||||||
myP0IncEvent = Event::PaddleThreeDecrease;
|
if(swapdir)
|
||||||
myP1DecEvent = Event::PaddleTwoIncrease;
|
{
|
||||||
myP1IncEvent = Event::PaddleTwoDecrease;
|
swapEvents(myP0DecEvent, myP0IncEvent);
|
||||||
}
|
swapEvents(myP1DecEvent, myP1IncEvent);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// The following are independent of whether or not the port
|
// The following are independent of whether or not the port
|
||||||
|
@ -179,6 +116,16 @@ Paddles::Paddles(Jack jack, const Event& event, const System& system,
|
||||||
setPin(DigitalPin::Six, true);
|
setPin(DigitalPin::Six, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
void Paddles::swapEvents(Event::Type& event1, Event::Type& event2)
|
||||||
|
{
|
||||||
|
Event::Type swappedEvent;
|
||||||
|
|
||||||
|
swappedEvent = event1;
|
||||||
|
event1 = event2;
|
||||||
|
event2 = swappedEvent;
|
||||||
|
}
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
void Paddles::update()
|
void Paddles::update()
|
||||||
{
|
{
|
||||||
|
@ -200,6 +147,31 @@ void Paddles::update()
|
||||||
// And to top it all off, we don't want one devices input to conflict
|
// And to top it all off, we don't want one devices input to conflict
|
||||||
// with the others ...
|
// with the others ...
|
||||||
|
|
||||||
|
if(!updateAnalogAxes())
|
||||||
|
{
|
||||||
|
updateMouse(firePressedP0, firePressedP1);
|
||||||
|
updateDigitalAxes();
|
||||||
|
|
||||||
|
// Only change state if the charge has actually changed
|
||||||
|
if(myCharge[1] != myLastCharge[1])
|
||||||
|
{
|
||||||
|
setPin(AnalogPin::Five, Int32(MAX_RESISTANCE * (myCharge[1] / double(TRIGMAX))));
|
||||||
|
myLastCharge[1] = myCharge[1];
|
||||||
|
}
|
||||||
|
if(myCharge[0] != myLastCharge[0])
|
||||||
|
{
|
||||||
|
setPin(AnalogPin::Nine, Int32(MAX_RESISTANCE * (myCharge[0] / double(TRIGMAX))));
|
||||||
|
myLastCharge[0] = myCharge[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setPin(DigitalPin::Four, !getAutoFireState(firePressedP0));
|
||||||
|
setPin(DigitalPin::Three, !getAutoFireStateP1(firePressedP1));
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
bool Paddles::updateAnalogAxes()
|
||||||
|
{
|
||||||
// Analog axis events from Stelladaptor-like devices
|
// Analog axis events from Stelladaptor-like devices
|
||||||
// These devices generate data in the range -32768 to 32767,
|
// These devices generate data in the range -32768 to 32767,
|
||||||
// so we have to scale appropriately
|
// so we have to scale appropriately
|
||||||
|
@ -207,10 +179,6 @@ void Paddles::update()
|
||||||
// we only process the first one we see (when it differs from
|
// we only process the first one we see (when it differs from
|
||||||
// previous values by a pre-defined amount)
|
// previous values by a pre-defined amount)
|
||||||
// Otherwise, it would always override input from digital and mouse
|
// Otherwise, it would always override input from digital and mouse
|
||||||
bool sa_changed = false;
|
|
||||||
int sa_xaxis = myEvent.get(myP0AxisValue);
|
|
||||||
int sa_yaxis = myEvent.get(myP1AxisValue);
|
|
||||||
int new_val;
|
|
||||||
|
|
||||||
static constexpr std::array<double, MAX_DEJITTER - MIN_DEJITTER + 1> bFac = {
|
static constexpr std::array<double, MAX_DEJITTER - MIN_DEJITTER + 1> bFac = {
|
||||||
// higher values mean more dejitter strength
|
// higher values mean more dejitter strength
|
||||||
|
@ -221,24 +189,28 @@ void Paddles::update()
|
||||||
static constexpr std::array<double, MAX_DEJITTER - MIN_DEJITTER + 1> dFac = {
|
static constexpr std::array<double, MAX_DEJITTER - MIN_DEJITTER + 1> dFac = {
|
||||||
// lower values mean more dejitter strength
|
// lower values mean more dejitter strength
|
||||||
1, // off
|
1, // off
|
||||||
1.0 / 181, 1.0 / 256, 1.0 / 362, 1.0 / 512, 1.0 / 724,
|
1.0 / 181, 1.0 / 256, 1.0 / 362, 1.0 / 512, 1.0 / 724,
|
||||||
1.0 / 1024, 1.0 / 1448, 1.0 / 2048, 1.0 / 2896, 1.0 / 4096
|
1.0 / 1024, 1.0 / 1448, 1.0 / 2048, 1.0 / 2896, 1.0 / 4096
|
||||||
};
|
};
|
||||||
const double baseFactor = bFac[DEJITTER_BASE];
|
const double baseFactor = bFac[DEJITTER_BASE];
|
||||||
const double diffFactor = dFac[DEJITTER_DIFF];
|
const double diffFactor = dFac[DEJITTER_DIFF];
|
||||||
|
|
||||||
|
int sa_xaxis = myEvent.get(myP0AxisValue);
|
||||||
|
int sa_yaxis = myEvent.get(myP1AxisValue);
|
||||||
|
bool sa_changed = false;
|
||||||
|
|
||||||
if(abs(myLastAxisX - sa_xaxis) > 10)
|
if(abs(myLastAxisX - sa_xaxis) > 10)
|
||||||
{
|
{
|
||||||
// dejitter, suppress small changes only
|
// dejitter, suppress small changes only
|
||||||
double dejitter = std::pow(baseFactor, abs(sa_xaxis - myLastAxisX) * diffFactor);
|
double dejitter = std::pow(baseFactor, abs(sa_xaxis - myLastAxisX) * diffFactor);
|
||||||
new_val = sa_xaxis * (1 - dejitter) + myLastAxisX * dejitter;
|
int new_val = sa_xaxis * (1 - dejitter) + myLastAxisX * dejitter;
|
||||||
|
|
||||||
// only use new dejittered value for larger differences
|
// only use new dejittered value for larger differences
|
||||||
if(abs(new_val - sa_xaxis) > 10)
|
if(abs(new_val - sa_xaxis) > 10)
|
||||||
sa_xaxis = new_val;
|
sa_xaxis = new_val;
|
||||||
|
|
||||||
setPin(AnalogPin::Nine, Int32(MAX_RESISTANCE *
|
setPin(AnalogPin::Nine, Int32(MAX_RESISTANCE *
|
||||||
(BSPF::clamp(32768 - Int32(Int32(sa_xaxis) * SENSITIVITY + XCENTER), 0, 65536) / 65536.0)));
|
(BSPF::clamp(32768 - Int32(Int32(sa_xaxis) * SENSITIVITY + XCENTER), 0, 65536) / 65536.0)));
|
||||||
sa_changed = true;
|
sa_changed = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -246,29 +218,33 @@ void Paddles::update()
|
||||||
{
|
{
|
||||||
// dejitter, suppress small changes only
|
// dejitter, suppress small changes only
|
||||||
double dejitter = std::pow(baseFactor, abs(sa_yaxis - myLastAxisY) * diffFactor);
|
double dejitter = std::pow(baseFactor, abs(sa_yaxis - myLastAxisY) * diffFactor);
|
||||||
new_val = sa_yaxis * (1 - dejitter) + myLastAxisY * dejitter;
|
int new_val = sa_yaxis * (1 - dejitter) + myLastAxisY * dejitter;
|
||||||
|
|
||||||
// only use new dejittered value for larger differences
|
// only use new dejittered value for larger differences
|
||||||
if (abs(new_val - sa_yaxis) > 10)
|
if(abs(new_val - sa_yaxis) > 10)
|
||||||
sa_yaxis = new_val;
|
sa_yaxis = new_val;
|
||||||
|
|
||||||
setPin(AnalogPin::Five, Int32(MAX_RESISTANCE *
|
setPin(AnalogPin::Five, Int32(MAX_RESISTANCE *
|
||||||
(BSPF::clamp(32768 - Int32(Int32(sa_yaxis) * SENSITIVITY + YCENTER), 0, 65536) / 65536.0)));
|
(BSPF::clamp(32768 - Int32(Int32(sa_yaxis) * SENSITIVITY + YCENTER), 0, 65536) / 65536.0)));
|
||||||
sa_changed = true;
|
sa_changed = true;
|
||||||
}
|
}
|
||||||
myLastAxisX = sa_xaxis;
|
myLastAxisX = sa_xaxis;
|
||||||
myLastAxisY = sa_yaxis;
|
myLastAxisY = sa_yaxis;
|
||||||
if(sa_changed)
|
|
||||||
return;
|
|
||||||
|
|
||||||
|
return sa_changed;
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
void Paddles::updateMouse(bool& firePressedP0, bool& firePressedP1)
|
||||||
|
{
|
||||||
// Mouse motion events give relative movement
|
// Mouse motion events give relative movement
|
||||||
// That is, they're only relevant if they're non-zero
|
// That is, they're only relevant if they're non-zero
|
||||||
if(myMPaddleID > -1)
|
if(myMPaddleID > -1)
|
||||||
{
|
{
|
||||||
// We're in auto mode, where a single axis is used for one paddle only
|
// We're in auto mode, where a single axis is used for one paddle only
|
||||||
myCharge[myMPaddleID] = BSPF::clamp(myCharge[myMPaddleID] -
|
myCharge[myMPaddleID] = BSPF::clamp(myCharge[myMPaddleID] -
|
||||||
(myEvent.get(myAxisMouseMotion) * MOUSE_SENSITIVITY),
|
(myEvent.get(myAxisMouseMotion) * MOUSE_SENSITIVITY),
|
||||||
TRIGMIN, TRIGRANGE);
|
TRIGMIN, TRIGRANGE);
|
||||||
if(myMPaddleID == 0)
|
if(myMPaddleID == 0)
|
||||||
firePressedP0 = firePressedP0
|
firePressedP0 = firePressedP0
|
||||||
|| myEvent.get(Event::MouseButtonLeftValue)
|
|| myEvent.get(Event::MouseButtonLeftValue)
|
||||||
|
@ -285,9 +261,8 @@ void Paddles::update()
|
||||||
if(myMPaddleIDX > -1)
|
if(myMPaddleIDX > -1)
|
||||||
{
|
{
|
||||||
myCharge[myMPaddleIDX] = BSPF::clamp(myCharge[myMPaddleIDX] -
|
myCharge[myMPaddleIDX] = BSPF::clamp(myCharge[myMPaddleIDX] -
|
||||||
(myEvent.get(Event::MouseAxisXMove) * MOUSE_SENSITIVITY),
|
(myEvent.get(Event::MouseAxisXMove) * MOUSE_SENSITIVITY),
|
||||||
TRIGMIN, TRIGRANGE);
|
TRIGMIN, TRIGRANGE);
|
||||||
|
|
||||||
if(myMPaddleIDX == 0)
|
if(myMPaddleIDX == 0)
|
||||||
firePressedP0 = firePressedP0
|
firePressedP0 = firePressedP0
|
||||||
|| myEvent.get(Event::MouseButtonLeftValue);
|
|| myEvent.get(Event::MouseButtonLeftValue);
|
||||||
|
@ -298,9 +273,8 @@ void Paddles::update()
|
||||||
if(myMPaddleIDY > -1)
|
if(myMPaddleIDY > -1)
|
||||||
{
|
{
|
||||||
myCharge[myMPaddleIDY] = BSPF::clamp(myCharge[myMPaddleIDY] -
|
myCharge[myMPaddleIDY] = BSPF::clamp(myCharge[myMPaddleIDY] -
|
||||||
(myEvent.get(Event::MouseAxisYMove) * MOUSE_SENSITIVITY),
|
(myEvent.get(Event::MouseAxisYMove) * MOUSE_SENSITIVITY),
|
||||||
TRIGMIN, TRIGRANGE);
|
TRIGMIN, TRIGRANGE);
|
||||||
|
|
||||||
if(myMPaddleIDY == 0)
|
if(myMPaddleIDY == 0)
|
||||||
firePressedP0 = firePressedP0
|
firePressedP0 = firePressedP0
|
||||||
|| myEvent.get(Event::MouseButtonRightValue);
|
|| myEvent.get(Event::MouseButtonRightValue);
|
||||||
|
@ -309,9 +283,11 @@ void Paddles::update()
|
||||||
|| myEvent.get(Event::MouseButtonRightValue);
|
|| myEvent.get(Event::MouseButtonRightValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
setPin(DigitalPin::Four, !getAutoFireState(firePressedP0));
|
}
|
||||||
setPin(DigitalPin::Three, !getAutoFireStateP1(firePressedP1));
|
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
void Paddles::updateDigitalAxes()
|
||||||
|
{
|
||||||
// Finally, consider digital input, where movement happens
|
// Finally, consider digital input, where movement happens
|
||||||
// until a digital event is released
|
// until a digital event is released
|
||||||
if(myKeyRepeat0)
|
if(myKeyRepeat0)
|
||||||
|
@ -354,15 +330,6 @@ void Paddles::update()
|
||||||
if((myCharge[myAxisDigitalOne] + myPaddleRepeat1) < TRIGRANGE)
|
if((myCharge[myAxisDigitalOne] + myPaddleRepeat1) < TRIGRANGE)
|
||||||
myCharge[myAxisDigitalOne] += myPaddleRepeat1;
|
myCharge[myAxisDigitalOne] += myPaddleRepeat1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only change state if the charge has actually changed
|
|
||||||
if(myCharge[1] != myLastCharge[1])
|
|
||||||
setPin(AnalogPin::Five, Int32(MAX_RESISTANCE * (myCharge[1] / double(TRIGMAX))));
|
|
||||||
if(myCharge[0] != myLastCharge[0])
|
|
||||||
setPin(AnalogPin::Nine, Int32(MAX_RESISTANCE * (myCharge[0] / double(TRIGMAX))));
|
|
||||||
|
|
||||||
myLastCharge[1] = myCharge[1];
|
|
||||||
myLastCharge[0] = myCharge[0];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
|
|
@ -197,6 +197,26 @@ class Paddles : public Controller
|
||||||
static int DEJITTER_BASE, DEJITTER_DIFF;
|
static int DEJITTER_BASE, DEJITTER_DIFF;
|
||||||
static int MOUSE_SENSITIVITY;
|
static int MOUSE_SENSITIVITY;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Swap two events.
|
||||||
|
*/
|
||||||
|
void swapEvents(Event::Type& event1, Event::Type& event2);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Update the axes pin state according to the events currently set.
|
||||||
|
*/
|
||||||
|
bool updateAnalogAxes();
|
||||||
|
|
||||||
|
/**
|
||||||
|
Update the entire state according to mouse events currently set.
|
||||||
|
*/
|
||||||
|
void updateMouse(bool& firePressedP0, bool& firePressedP1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Update the axes pin state according to the keyboard events currently set.
|
||||||
|
*/
|
||||||
|
void updateDigitalAxes();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Following constructors and assignment operators not supported
|
// Following constructors and assignment operators not supported
|
||||||
Paddles() = delete;
|
Paddles() = delete;
|
||||||
|
|
Loading…
Reference in New Issue