metaforce/Runtime/Input/CInputGenerator.hpp

246 lines
7.3 KiB
C++
Raw Normal View History

2018-10-07 03:42:33 +00:00
#pragma once
2015-08-18 05:54:43 +00:00
2015-08-29 01:30:47 +00:00
#include <atomic>
2015-08-28 00:11:31 +00:00
#include <boo/boo.hpp>
2015-08-29 01:30:47 +00:00
#include "CFinalInput.hpp"
#include "CKeyboardMouseController.hpp"
2015-08-28 00:11:31 +00:00
2016-03-04 23:04:53 +00:00
namespace urde
2015-08-18 05:54:43 +00:00
{
2015-08-29 01:30:47 +00:00
class CArchitectureQueue;
2015-08-18 05:54:43 +00:00
2017-02-18 02:19:50 +00:00
enum class EIOPort
{
Zero,
One,
Two,
Three
};
enum class EMotorState
{
Stop,
Rumble,
StopHard
};
2015-08-29 01:30:47 +00:00
class CInputGenerator : public boo::DeviceFinder
2015-08-18 05:54:43 +00:00
{
2015-11-21 01:16:07 +00:00
enum class EStatusChange
2015-08-29 01:30:47 +00:00
{
2015-11-21 01:16:07 +00:00
NoChange = 0,
Connected = 1,
Disconnected = 2
2015-08-29 01:30:47 +00:00
};
/* When the sticks are used as logical (digital) input,
* these thresholds determine the vector magnitude indicating
* the logical state */
float m_leftDiv;
float m_rightDiv;
2016-02-20 06:45:36 +00:00
CKeyboardMouseControllerData m_data;
CFinalInput m_lastUpdate;
const CFinalInput& getFinalInput(unsigned idx, float dt)
{
m_lastUpdate = CFinalInput(idx, dt, m_data, m_lastUpdate);
return m_lastUpdate;
}
2016-02-21 06:34:42 +00:00
bool m_firstFrame = true;
2015-08-29 01:30:47 +00:00
public:
CInputGenerator(float leftDiv, float rightDiv)
2018-10-07 02:59:17 +00:00
: boo::DeviceFinder({dev_typeid(DolphinSmashAdapter)}),
2015-08-29 01:30:47 +00:00
m_leftDiv(leftDiv),
m_rightDiv(rightDiv) {}
2017-12-16 00:19:15 +00:00
~CInputGenerator()
{
if (smashAdapter)
smashAdapter->setCallback(nullptr);
}
2015-08-29 01:30:47 +00:00
/* Keyboard and mouse events are delivered on the main game
* thread as part of the app's main event loop. The OS is responsible
* for buffering events in its own way, then boo flushes the buffer
* at the start of each frame, invoking these methods. No atomic locking
* is necessary, only absolute state tracking. */
2016-02-20 06:45:36 +00:00
void mouseDown(const boo::SWindowCoord&, boo::EMouseButton button, boo::EModifierKey)
{
m_data.m_mouseButtons[int(button)] = true;
}
void mouseUp(const boo::SWindowCoord&, boo::EMouseButton button, boo::EModifierKey)
{
m_data.m_mouseButtons[int(button)] = false;
}
void mouseMove(const boo::SWindowCoord& coord)
{
m_data.m_mouseCoord = coord;
}
void scroll(const boo::SWindowCoord&, const boo::SScrollDelta& scroll)
{
m_data.m_accumScroll += scroll;
}
2015-08-28 00:11:31 +00:00
2016-02-20 06:45:36 +00:00
void charKeyDown(unsigned long charCode, boo::EModifierKey, bool)
{
charCode = tolower(charCode);
if (charCode > 255)
return;
m_data.m_charKeys[charCode] = true;
}
void charKeyUp(unsigned long charCode, boo::EModifierKey mods)
{
charCode = tolower(charCode);
if (charCode > 255)
return;
m_data.m_charKeys[charCode] = false;
}
void specialKeyDown(boo::ESpecialKey key, boo::EModifierKey, bool)
{
m_data.m_specialKeys[int(key)] = true;
}
void specialKeyUp(boo::ESpecialKey key, boo::EModifierKey)
{
m_data.m_specialKeys[int(key)] = false;
}
void modKeyDown(boo::EModifierKey mod, bool)
{
m_data.m_modMask = m_data.m_modMask | mod;
}
void modKeyUp(boo::EModifierKey mod)
{
m_data.m_modMask = m_data.m_modMask & ~mod;
}
2015-08-28 00:11:31 +00:00
2016-02-20 06:45:36 +00:00
void reset()
{
m_data.m_accumScroll.zeroOut();
}
2015-08-28 00:11:31 +00:00
2015-08-29 01:30:47 +00:00
/* Input via the smash adapter is received asynchronously on a USB
* report thread. This class atomically exchanges that data to the
* game thread as needed */
2015-08-28 00:11:31 +00:00
struct DolphinSmashAdapterCallback : boo::IDolphinSmashAdapterCallback
{
2015-08-29 01:30:47 +00:00
std::atomic<EStatusChange> m_statusChanges[4];
bool m_connected[4] = {};
boo::DolphinControllerState m_states[4];
std::mutex m_stateLock;
2015-08-28 00:11:31 +00:00
void controllerConnected(unsigned idx, boo::EDolphinControllerType)
{
2015-08-29 01:30:47 +00:00
/* Controller thread */
2015-11-21 01:16:07 +00:00
m_statusChanges[idx].store(EStatusChange::Connected);
2015-08-28 00:11:31 +00:00
}
void controllerDisconnected(unsigned idx)
2015-08-28 00:11:31 +00:00
{
2015-08-29 01:30:47 +00:00
/* Controller thread */
std::unique_lock<std::mutex> lk(m_stateLock);
2015-11-21 01:16:07 +00:00
m_statusChanges[idx].store(EStatusChange::Disconnected);
2015-08-29 01:30:47 +00:00
m_states[idx].reset();
2015-08-28 00:11:31 +00:00
}
void controllerUpdate(unsigned idx, boo::EDolphinControllerType,
const boo::DolphinControllerState& state)
{
2015-08-29 01:30:47 +00:00
/* Controller thread */
std::unique_lock<std::mutex> lk(m_stateLock);
m_states[idx] = state;
}
CFinalInput m_lastUpdates[4];
const CFinalInput& getFinalInput(unsigned idx, float dt,
float leftDiv, float rightDiv)
{
/* Game thread */
std::unique_lock<std::mutex> lk(m_stateLock);
boo::DolphinControllerState state = m_states[idx];
lk.unlock();
state.clamp(); /* PADClamp equivalent */
m_lastUpdates[idx] = CFinalInput(idx, dt, state, m_lastUpdates[idx], leftDiv, rightDiv);
return m_lastUpdates[idx];
}
EStatusChange getStatusChange(unsigned idx, bool& connected)
{
/* Game thread */
2015-11-21 01:16:07 +00:00
EStatusChange ch = m_statusChanges[idx].exchange(EStatusChange::NoChange);
if (ch == EStatusChange::Connected)
2015-08-29 01:30:47 +00:00
m_connected[idx] = true;
2015-11-21 01:16:07 +00:00
else if (ch == EStatusChange::Disconnected)
2015-08-29 01:30:47 +00:00
m_connected[idx] = false;
connected = m_connected[idx];
return ch;
}
} m_dolphinCb;
/* Device connection/disconnection events are handled on a separate thread
* using the relevant OS API. This thread blocks in a loop until an event is
* received. Device pointers should only be manipulated by this thread using
* the deviceConnected() and deviceDisconnected() callbacks. */
std::shared_ptr<boo::DolphinSmashAdapter> smashAdapter;
2015-08-29 01:30:47 +00:00
void deviceConnected(boo::DeviceToken& tok)
2015-08-28 00:11:31 +00:00
{
2015-08-29 01:30:47 +00:00
/* Device listener thread */
if (!smashAdapter)
{
2018-10-07 02:59:17 +00:00
auto dev = tok.openAndGetDevice();
if (dev && dev->getTypeHash() == dev_typeid(DolphinSmashAdapter))
{
smashAdapter = std::static_pointer_cast<boo::DolphinSmashAdapter>(tok.openAndGetDevice());
smashAdapter->setCallback(&m_dolphinCb);
2018-10-07 02:59:17 +00:00
}
2015-08-28 00:11:31 +00:00
}
2015-08-29 01:30:47 +00:00
}
void deviceDisconnected(boo::DeviceToken&, boo::DeviceBase* device)
{
if (smashAdapter.get() == device)
smashAdapter.reset();
2015-08-29 01:30:47 +00:00
}
2017-02-18 02:19:50 +00:00
void SetMotorState(EIOPort port, EMotorState state)
{
if (smashAdapter)
{
switch (state)
{
case EMotorState::Stop:
smashAdapter->stopRumble(unsigned(port));
break;
case EMotorState::Rumble:
smashAdapter->startRumble(unsigned(port));
break;
case EMotorState::StopHard:
smashAdapter->stopRumble(unsigned(port), true);
break;
}
}
}
2018-01-26 09:48:42 +00:00
void ControlAllMotors(const EMotorState* states)
{
if (smashAdapter)
{
for (int i=0 ; i<4 ; ++i)
{
switch (states[i])
{
case EMotorState::Stop:
smashAdapter->stopRumble(i);
break;
case EMotorState::Rumble:
smashAdapter->startRumble(i);
break;
case EMotorState::StopHard:
smashAdapter->stopRumble(i, true);
break;
}
}
}
}
2015-08-29 01:30:47 +00:00
/* This is where the game thread enters */
void Update(float dt, CArchitectureQueue& queue);
2015-08-28 00:11:31 +00:00
2015-08-18 05:54:43 +00:00
};
}