A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KPadController.hh
1#pragma once
2
3#include "game/system/GhostFile.hh"
4
5#include <egg/math/Vector.hh>
6
7namespace System {
8
12[[nodiscard]] static constexpr f32 RawStickToState(u8 rawStick) {
13 return (static_cast<f32>(rawStick) - 7.0f) / 7.0f;
14}
15
16enum class ControlSource {
17 Unknown = -1,
18 Core = 0, // WiiMote
19 Freestyle = 1, // WiiMote + Nunchuk
20 Classic = 2,
21 Gamecube = 3,
22 Ghost = 4,
23 AI = 5,
24 Host = 6, // Added in Kinoko, represents an external program
25};
26
27enum class Trick {
28 None = 0,
29 Up = 1,
30 Down = 2,
31 Left = 3,
32 Right = 4,
33};
34
38 virtual ~RaceInputState() {}
39
40 void reset();
41
42 [[nodiscard]] bool isValid() const;
43
48 [[nodiscard]] bool isButtonsValid() const {
49 return !(buttons & ~0xf);
50 }
51
52 [[nodiscard]] bool isStickValid(f32 stick) const;
53
56 [[nodiscard]] bool isTrickValid() const {
57 switch (trick) {
58 case Trick::None:
59 case Trick::Up:
60 case Trick::Down:
61 case Trick::Left:
62 case Trick::Right:
63 return true;
64 default:
65 return false;
66 }
67 }
68
69 [[nodiscard]] bool accelerate() const {
70 return !!(buttons & 0x1);
71 }
72
73 [[nodiscard]] bool brake() const {
74 return !!(buttons & 0x2);
75 }
76
77 [[nodiscard]] bool item() const {
78 return !!(buttons & 0x4);
79 }
80
81 [[nodiscard]] bool drift() const {
82 return !!(buttons & 0x8);
83 }
84
85 [[nodiscard]] bool trickUp() const {
86 return trick == Trick::Up;
87 }
88
89 [[nodiscard]] bool trickDown() const {
90 return trick == Trick::Down;
91 }
92
93 u16 buttons;
94 u16 buttonsRaw;
95 EGG::Vector2f stick;
96 u8 stickXRaw;
97 u8 stickYRaw;
98 Trick trick;
99 u8 trickRaw;
100};
101
106
107 [[nodiscard]] virtual u8 readFrame();
108
110 [[nodiscard]] virtual bool readIsNewSequence() const {
111 return readSequenceFrames >= (currentSequence & 0xFF);
112 }
113
115 [[nodiscard]] virtual u8 readVal() const {
116 return currentSequence >> 8;
117 }
118
119 EGG::RamStream buffer;
120 u32 currentSequence;
121 u16 readSequenceFrames;
122 u32 state;
123};
124
141
152
162
164 [[nodiscard]] bool readIsNewSequence() const override {
165 u16 duration = currentSequence & 0xFF;
166 duration += 256 * (currentSequence >> 8 & 0xF);
167 return duration <= readSequenceFrames;
168 }
169
171 [[nodiscard]] u8 readVal() const override {
172 return currentSequence >> 0x8 & ~0x80;
173 }
174};
175
178public:
180 virtual ~KPadController() {}
181
183 [[nodiscard]] virtual ControlSource controlSource() const {
184 return ControlSource::Unknown;
185 }
186
187 virtual void reset(bool /*driftIsAuto*/) {}
188 virtual void calcImpl() {}
189
190 void calc();
191
192 [[nodiscard]] const RaceInputState &raceInputState() const {
193 return m_raceInputState;
194 }
195
197 void setDriftIsAuto(bool driftIsAuto) {
198 m_driftIsAuto = driftIsAuto;
199 }
200
201 [[nodiscard]] bool driftIsAuto() const {
202 return m_driftIsAuto;
203 }
204
205protected:
209};
210
214public:
216 ~KPadGhostController() override;
217
219 [[nodiscard]] ControlSource controlSource() const override {
220 return ControlSource::Ghost;
221 }
222
223 void reset(bool driftIsAuto) override;
224
225 void readGhostBuffer(const u8 *buffer, bool driftIsAuto);
226
227 void calcImpl() override;
228
229 void setAcceptingInputs(bool set) {
230 m_acceptingInputs = set;
231 }
232
233private:
234 const u8 *m_ghostBuffer;
235 std::array<KPadGhostButtonsStream *, 3> m_buttonsStreams;
236 bool m_acceptingInputs;
237};
238
242public:
244 ~KPadHostController() override;
245
246 [[nodiscard]] ControlSource controlSource() const override {
247 return ControlSource::Host;
248 }
249
250 void reset(bool driftIsAuto) override;
251
256 bool setInputs(const RaceInputState &state) {
257 return setInputs(state.buttons, state.stick, state.trick);
258 }
259
265 bool setInputs(u16 buttons, const EGG::Vector2f &stick, Trick trick) {
266 m_raceInputState.buttons = buttons;
267 m_raceInputState.stick = stick;
268 m_raceInputState.trick = trick;
269
270 return m_raceInputState.isValid();
271 }
272
279 bool setInputs(u16 buttons, f32 stickX, f32 stickY, Trick trick) {
280 m_raceInputState.buttons = buttons;
281 m_raceInputState.stick.x = stickX;
282 m_raceInputState.stick.y = stickY;
283 m_raceInputState.trick = trick;
284
285 return m_raceInputState.isValid();
286 }
287
295 bool setInputsRawStick(u16 buttons, u8 stickXRaw, u8 stickYRaw, Trick trick) {
296 return setInputs(buttons, RawStickToState(stickXRaw), RawStickToState(stickYRaw), trick);
297 }
298
306 bool setInputsRawStickZeroCenter(u16 buttons, s8 stickXRaw, s8 stickYRaw, Trick trick) {
307 return setInputsRawStick(buttons, stickXRaw + 7, stickYRaw + 7, trick);
308 }
309};
310
311class KPad {
312public:
313 KPad();
314 ~KPad();
315
316 void calc();
317 void reset();
318
319 [[nodiscard]] const RaceInputState &currentState() const {
320 return m_currentInputState;
321 }
322
323 [[nodiscard]] const RaceInputState &lastState() const {
324 return m_lastInputState;
325 }
326
327 [[nodiscard]] bool driftIsAuto() const {
328 return m_controller->driftIsAuto();
329 }
330
331protected:
332 KPadController *m_controller;
333 RaceInputState m_currentInputState;
335};
336
338class KPadPlayer : public KPad {
339public:
340 KPadPlayer();
341 ~KPadPlayer();
342
343 void setGhostController(KPadGhostController *controller, const u8 *inputs, bool driftIsAuto);
344 void setHostController(KPadHostController *controller, bool driftIsAuto);
345
346 void startGhostProxy();
347 void endGhostProxy();
348
349private:
350 u8 m_ghostBuffer[RKG_UNCOMPRESSED_INPUT_DATA_SECTION_SIZE];
351};
352
353} // namespace System
A stream of data stored in memory.
Definition Stream.hh:64
An abstraction for a controller object. It is associated with an input state.
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 readGhostBuffer(const u8 *buffer, bool driftIsAuto)
Reads in the raw input data section from the ghost RKG file.
The abstraction of a controller object but for external usage.
bool setInputs(u16 buttons, const EGG::Vector2f &stick, Trick trick)
Sets the inputs of the controller.
bool setInputs(const RaceInputState &state)
Sets the inputs of the controller.
bool setInputs(u16 buttons, f32 stickX, f32 stickY, Trick trick)
Sets the inputs of the controller.
bool setInputsRawStickZeroCenter(u16 buttons, s8 stickXRaw, s8 stickYRaw, Trick trick)
Sets the inputs of the controller.
bool setInputsRawStick(u16 buttons, u8 stickXRaw, u8 stickYRaw, Trick trick)
Sets the inputs of the controller.
A specialized KPad for player input, as opposed to CPU players for example.
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.
Represents the host application.
Definition Option.cc:5
High-level handling for generic system operations, such as input reading, race configuration,...
Definition CourseMap.cc:5
static constexpr f32 RawStickToState(u8 rawStick)
Converts a raw stick input into an input usable by the state.
A 2D float vector.
Definition Vector.hh:12
Represents a stream of button inputs from a ghost file.
virtual u8 readFrame()
Reads the data from the corresponding tuple in the buffer.
A specialized stream for the analog stick. Direction tuples take the following form:
A specialized stream for button presses (not tricks).
A specialized stream for D-Pad inputs for tricking and wheeling. Trick tuples take the following form...
Represents a set of controller inputs.
bool isButtonsValid() const
Checks if there are any invalid buttons.
bool isValid() const
Checks if the input state is valid.
bool isStickValid(f32 stick) const
Checks if the stick values are within the domain of the physics engine.
bool isTrickValid() const
Checks if the trick input is valid.