1#include "KPadController.hh"
8KPadController::KPadController() : m_connected(false) {}
11void KPadController::calc() {
16KPadGhostController::KPadGhostController() : m_acceptingInputs(false) {
17 m_buttonsStreams[0] =
new KPadGhostFaceButtonsStream;
18 m_buttonsStreams[1] =
new KPadGhostDirectionButtonsStream;
19 m_buttonsStreams[2] =
new KPadGhostTrickButtonsStream;
23KPadGhostController::~KPadGhostController() =
default;
26void KPadGhostController::reset(
bool driftIsAuto) {
27 m_driftIsAuto = driftIsAuto;
28 m_raceInputState.reset();
30 for (
auto &stream : m_buttonsStreams) {
31 stream->currentSequence = 0;
32 stream->readSequenceFrames = 0;
36 m_acceptingInputs =
false;
54void KPadGhostController::readGhostBuffer(
const u8 *buffer,
bool driftIsAuto) {
55 constexpr u32 SEQUENCE_SIZE = 0x2;
57 m_ghostBuffer = buffer;
58 m_driftIsAuto = driftIsAuto;
62 u16 faceCount = stream.read_u16();
63 u16 directionCount = stream.read_u16();
64 u16 trickCount = stream.read_u16();
68 m_buttonsStreams[0]->buffer = stream.
split(faceCount * SEQUENCE_SIZE);
69 m_buttonsStreams[1]->buffer = stream.
split(directionCount * SEQUENCE_SIZE);
70 m_buttonsStreams[2]->buffer = stream.
split(trickCount * SEQUENCE_SIZE);
74void KPadGhostController::calcImpl() {
75 if (!m_ghostBuffer || !m_acceptingInputs) {
79 m_raceInputState.buttons = m_buttonsStreams[0]->readFrame();
80 u8 sticks = m_buttonsStreams[1]->readFrame();
81 m_raceInputState.stickXRaw = sticks >> 4 & 0xF;
82 m_raceInputState.stickYRaw = sticks & 0xF;
83 m_raceInputState.stick =
EGG::Vector2f(RawStickToState(m_raceInputState.stickXRaw),
84 RawStickToState(m_raceInputState.stickYRaw));
85 m_raceInputState.trickRaw = m_buttonsStreams[2]->readFrame();
87 switch (m_raceInputState.trickRaw >> 4) {
89 m_raceInputState.trick = Trick::Up;
92 m_raceInputState.trick = Trick::Down;
95 m_raceInputState.trick = Trick::Left;
98 m_raceInputState.trick = Trick::Right;
101 m_raceInputState.trick = Trick::None;
106RaceInputState::RaceInputState() {
111void RaceInputState::reset() {
114 stick = EGG::Vector2f::zero;
123bool RaceInputState::isValid()
const {
124 if (!isButtonsValid()) {
128 if (!isStickValid(stick.x) || !isStickValid(stick.y)) {
132 if (!isTrickValid()) {
143bool RaceInputState::isStickValid(f32 stick)
const {
144 if (stick > 1.0f || stick < -1.0f) {
148 for (
size_t i = 0; i <= 14; ++i) {
149 auto cond = stick <=> RawStickToState(i);
150 ASSERT(cond != std::partial_ordering::unordered);
152 if (cond == std::partial_ordering::equivalent) {
154 }
else if (cond == std::partial_ordering::less) {
163KPadGhostButtonsStream::KPadGhostButtonsStream()
164 : currentSequence(std::numeric_limits<u32>::max()), state(2) {}
166KPadGhostButtonsStream::~KPadGhostButtonsStream() =
default;
175 if (currentSequence == std::numeric_limits<u32>::max()) {
176 readSequenceFrames = 0;
177 currentSequence = buffer.read_u16();
179 if (readIsNewSequence()) {
180 readSequenceFrames = 0;
181 currentSequence = buffer.read_u16();
185 ++readSequenceFrames;
191 if (buffer.eof() && readIsNewSequence()) {
198KPadGhostFaceButtonsStream::KPadGhostFaceButtonsStream() =
default;
200KPadGhostFaceButtonsStream::~KPadGhostFaceButtonsStream() =
default;
202KPadGhostDirectionButtonsStream::KPadGhostDirectionButtonsStream() =
default;
204KPadGhostDirectionButtonsStream::~KPadGhostDirectionButtonsStream() =
default;
206KPadGhostTrickButtonsStream::KPadGhostTrickButtonsStream() =
default;
208KPadGhostTrickButtonsStream::~KPadGhostTrickButtonsStream() =
default;
214KPadHostController::KPadHostController() =
default;
216KPadHostController::~KPadHostController() =
default;
218void KPadHostController::reset(
bool driftIsAuto) {
229KPad::KPad() : m_controller(nullptr) {
234KPad::~KPad() =
default;
239 m_currentInputState = m_controller->raceInputState();
245 m_controller->reset(m_controller->driftIsAuto());
250KPadPlayer::KPadPlayer() =
default;
253KPadPlayer::~KPadPlayer() =
default;
256void KPadPlayer::setGhostController(KPadGhostController *controller,
const u8 *inputs,
258 m_controller = controller;
261 memcpy(m_ghostBuffer, inputs, RKG_UNCOMPRESSED_INPUT_DATA_SECTION_SIZE);
264 controller->readGhostBuffer(m_ghostBuffer, driftIsAuto);
267void KPadPlayer::setHostController(KPadHostController *controller,
bool driftIsAuto) {
268 m_controller = controller;
269 m_controller->setDriftIsAuto(driftIsAuto);
274 if (!m_controller || m_controller->controlSource() != ControlSource::Ghost) {
279 ghostController->setAcceptingInputs(
true);
284 if (!m_controller || m_controller->controlSource() != ControlSource::Ghost) {
289 ghostController->setAcceptingInputs(
false);
A stream of data stored in memory.
RamStream split(u32 size)
Splits the current stream into two.
bool m_connected
Whether the controller is active.
bool m_driftIsAuto
True for auto transmission, false for manual.
RaceInputState m_raceInputState
The current inputs from this controller.
The abstraction of a controller object but for ghost playback.
void startGhostProxy()
Signals to start reading ghost data after fade-in.
void endGhostProxy()
Signals to stop reading ghost data after race completion.
RaceInputState m_lastInputState
Used to determine changes in input state.
High-level handling for generic system operations, such as input reading, race configuration,...