1
0
mirror of https://github.com/TomHarte/CLK.git synced 2025-01-12 00:30:31 +00:00

Moves the joystick class towards accepting analogue inputs.

This commit is contained in:
Thomas Harte 2018-06-10 20:45:52 -04:00
parent a1c60152d4
commit 3ea2a4ccb8
7 changed files with 109 additions and 76 deletions

View File

@ -25,11 +25,11 @@ class MultiJoystick: public Inputs::Joystick {
} }
} }
std::vector<DigitalInput> get_inputs() override { std::vector<Input> get_inputs() override {
std::vector<DigitalInput> inputs; std::vector<Input> inputs;
for(const auto &joystick: joysticks_) { for(const auto &joystick: joysticks_) {
std::vector<DigitalInput> joystick_inputs = joystick->get_inputs(); std::vector<Input> joystick_inputs = joystick->get_inputs();
for(const auto &input: joystick_inputs) { for(const auto &input: joystick_inputs) {
if(std::find(inputs.begin(), inputs.end(), input) != inputs.end()) { if(std::find(inputs.begin(), inputs.end(), input) != inputs.end()) {
inputs.push_back(input); inputs.push_back(input);
@ -40,9 +40,9 @@ class MultiJoystick: public Inputs::Joystick {
return inputs; return inputs;
} }
void set_digital_input(const DigitalInput &digital_input, bool is_active) override { void set_input(const Input &digital_input, bool is_active) override {
for(const auto &joystick: joysticks_) { for(const auto &joystick: joysticks_) {
joystick->set_digital_input(digital_input, is_active); joystick->set_input(digital_input, is_active);
} }
} }
void reset_all_inputs() override { void reset_all_inputs() override {

View File

@ -21,29 +21,57 @@ class Joystick {
public: public:
virtual ~Joystick() {} virtual ~Joystick() {}
struct DigitalInput { struct Input {
/// Defines the broad type of the input.
enum Type { enum Type {
Up, Down, Left, Right, Fire, // Half-axis inputs.
Up, Down, Left, Right,
// Full-axis inputs.
Horizontal, Vertical,
// Fire buttons.
Fire,
// Other labelled keys.
Key Key
} type; };
union { const Type type;
enum Precision {
Analogue, Digital
};
const Precision precision;
/*!
Holds extra information pertaining to the input.
@c Type::Key inputs declare the symbol printed on them.
All other types of input have an associated index, indicating whether they
are the zeroth, first, second, third, etc of those things. E.g. a joystick
may have two fire buttons, which will be buttons 0 and 1.
*/
union Info {
struct { struct {
int index; int index;
} control; } control;
struct { struct {
wchar_t symbol; wchar_t symbol;
} key; } key;
} info; };
Info info;
// TODO: Find a way to make the above safely const; may mean not using a union.
DigitalInput(Type type, int index = 0) : type(type) { Input(Type type, int index = 0, Precision precision = Precision::Digital) :
type(type),
precision(precision) {
info.control.index = index; info.control.index = index;
} }
DigitalInput(wchar_t symbol) : type(Key) { Input(wchar_t symbol) : type(Key), precision(Precision::Digital) {
info.key.symbol = symbol; info.key.symbol = symbol;
} }
bool operator == (const DigitalInput &rhs) { bool operator == (const Input &rhs) {
if(rhs.type != type) return false; if(rhs.type != type) return false;
if(rhs.precision != precision) return false;
if(rhs.type == Key) { if(rhs.type == Key) {
return rhs.info.key.symbol == info.key.symbol; return rhs.info.key.symbol == info.key.symbol;
} else { } else {
@ -52,13 +80,16 @@ class Joystick {
} }
}; };
virtual std::vector<DigitalInput> get_inputs() = 0; virtual std::vector<Input> get_inputs() = 0;
// Host interface. Note that the two set_inputs have logic to map
// between analogue and digital inputs; if you override
virtual void set_input(const Input &input, bool is_active) = 0;
virtual void set_input(const Input &input, float value) {}
// Host interface.
virtual void set_digital_input(const DigitalInput &digital_input, bool is_active) = 0;
virtual void reset_all_inputs() { virtual void reset_all_inputs() {
for(const auto &input: get_inputs()) { for(const auto &input: get_inputs()) {
set_digital_input(input, false); set_input(input, false);
} }
} }
}; };

View File

@ -42,25 +42,25 @@ class Joystick: public Inputs::Joystick {
Joystick(Bus *bus, std::size_t shift, std::size_t fire_tia_input) : Joystick(Bus *bus, std::size_t shift, std::size_t fire_tia_input) :
bus_(bus), shift_(shift), fire_tia_input_(fire_tia_input) {} bus_(bus), shift_(shift), fire_tia_input_(fire_tia_input) {}
std::vector<DigitalInput> get_inputs() override { std::vector<Input> get_inputs() override {
return { return {
DigitalInput(DigitalInput::Up), Input(Input::Up),
DigitalInput(DigitalInput::Down), Input(Input::Down),
DigitalInput(DigitalInput::Left), Input(Input::Left),
DigitalInput(DigitalInput::Right), Input(Input::Right),
DigitalInput(DigitalInput::Fire) Input(Input::Fire)
}; };
} }
void set_digital_input(const DigitalInput &digital_input, bool is_active) override { void set_input(const Input &digital_input, bool is_active) override {
switch(digital_input.type) { switch(digital_input.type) {
case DigitalInput::Up: bus_->mos6532_.update_port_input(0, 0x10 >> shift_, is_active); break; case Input::Up: bus_->mos6532_.update_port_input(0, 0x10 >> shift_, is_active); break;
case DigitalInput::Down: bus_->mos6532_.update_port_input(0, 0x20 >> shift_, is_active); break; case Input::Down: bus_->mos6532_.update_port_input(0, 0x20 >> shift_, is_active); break;
case DigitalInput::Left: bus_->mos6532_.update_port_input(0, 0x40 >> shift_, is_active); break; case Input::Left: bus_->mos6532_.update_port_input(0, 0x40 >> shift_, is_active); break;
case DigitalInput::Right: bus_->mos6532_.update_port_input(0, 0x80 >> shift_, is_active); break; case Input::Right: bus_->mos6532_.update_port_input(0, 0x80 >> shift_, is_active); break;
// TODO: latching // TODO: latching
case DigitalInput::Fire: case Input::Fire:
if(is_active) if(is_active)
bus_->tia_input_value_[fire_tia_input_] &= ~0x80; bus_->tia_input_value_[fire_tia_input_] &= ~0x80;
else else

View File

@ -34,28 +34,28 @@ namespace Vision {
class Joystick: public Inputs::Joystick { class Joystick: public Inputs::Joystick {
public: public:
std::vector<DigitalInput> get_inputs() override { std::vector<Input> get_inputs() override {
return { return {
DigitalInput(DigitalInput::Up), Input(Input::Up),
DigitalInput(DigitalInput::Down), Input(Input::Down),
DigitalInput(DigitalInput::Left), Input(Input::Left),
DigitalInput(DigitalInput::Right), Input(Input::Right),
DigitalInput(DigitalInput::Fire, 0), Input(Input::Fire, 0),
DigitalInput(DigitalInput::Fire, 1), Input(Input::Fire, 1),
DigitalInput('0'), DigitalInput('1'), DigitalInput('2'), Input('0'), Input('1'), Input('2'),
DigitalInput('3'), DigitalInput('4'), DigitalInput('5'), Input('3'), Input('4'), Input('5'),
DigitalInput('6'), DigitalInput('7'), DigitalInput('8'), Input('6'), Input('7'), Input('8'),
DigitalInput('9'), DigitalInput('*'), DigitalInput('#'), Input('9'), Input('*'), Input('#'),
}; };
} }
void set_digital_input(const DigitalInput &digital_input, bool is_active) override { void set_input(const Input &digital_input, bool is_active) override {
switch(digital_input.type) { switch(digital_input.type) {
default: return; default: return;
case DigitalInput::Key: case Input::Key:
if(!is_active) keypad_ |= 0xf; if(!is_active) keypad_ |= 0xf;
else { else {
uint8_t mask = 0xf; uint8_t mask = 0xf;
@ -78,11 +78,11 @@ class Joystick: public Inputs::Joystick {
} }
break; break;
case DigitalInput::Up: if(is_active) direction_ &= ~0x01; else direction_ |= 0x01; break; case Input::Up: if(is_active) direction_ &= ~0x01; else direction_ |= 0x01; break;
case DigitalInput::Right: if(is_active) direction_ &= ~0x02; else direction_ |= 0x02; break; case Input::Right: if(is_active) direction_ &= ~0x02; else direction_ |= 0x02; break;
case DigitalInput::Down: if(is_active) direction_ &= ~0x04; else direction_ |= 0x04; break; case Input::Down: if(is_active) direction_ &= ~0x04; else direction_ |= 0x04; break;
case DigitalInput::Left: if(is_active) direction_ &= ~0x08; else direction_ |= 0x08; break; case Input::Left: if(is_active) direction_ &= ~0x08; else direction_ |= 0x08; break;
case DigitalInput::Fire: case Input::Fire:
switch(digital_input.info.control.index) { switch(digital_input.info.control.index) {
default: break; default: break;
case 0: if(is_active) direction_ &= ~0x40; else direction_ |= 0x40; break; case 0: if(is_active) direction_ &= ~0x40; else direction_ |= 0x40; break;

View File

@ -263,25 +263,25 @@ class Joystick: public Inputs::Joystick {
user_port_via_port_handler_(user_port_via_port_handler), user_port_via_port_handler_(user_port_via_port_handler),
keyboard_via_port_handler_(keyboard_via_port_handler) {} keyboard_via_port_handler_(keyboard_via_port_handler) {}
std::vector<DigitalInput> get_inputs() override { std::vector<Input> get_inputs() override {
return { return {
DigitalInput(DigitalInput::Up), Input(Input::Up),
DigitalInput(DigitalInput::Down), Input(Input::Down),
DigitalInput(DigitalInput::Left), Input(Input::Left),
DigitalInput(DigitalInput::Right), Input(Input::Right),
DigitalInput(DigitalInput::Fire) Input(Input::Fire)
}; };
} }
void set_digital_input(const DigitalInput &digital_input, bool is_active) override { void set_input(const Input &digital_input, bool is_active) override {
JoystickInput mapped_input; JoystickInput mapped_input;
switch(digital_input.type) { switch(digital_input.type) {
default: return; default: return;
case DigitalInput::Up: mapped_input = Up; break; case Input::Up: mapped_input = Up; break;
case DigitalInput::Down: mapped_input = Down; break; case Input::Down: mapped_input = Down; break;
case DigitalInput::Left: mapped_input = Left; break; case Input::Left: mapped_input = Left; break;
case DigitalInput::Right: mapped_input = Right; break; case Input::Right: mapped_input = Right; break;
case DigitalInput::Fire: mapped_input = Fire; break; case Input::Fire: mapped_input = Fire; break;
} }
user_port_via_port_handler_.set_joystick_state(mapped_input, is_active); user_port_via_port_handler_.set_joystick_state(mapped_input, is_active);

View File

@ -252,19 +252,21 @@ struct SpeakerDelegate: public Outputs::Speaker::Speaker::Delegate, public LockP
@synchronized(self) { @synchronized(self) {
std::vector<std::unique_ptr<Inputs::Joystick>> &joysticks = joystick_machine->get_joysticks(); std::vector<std::unique_ptr<Inputs::Joystick>> &joysticks = joystick_machine->get_joysticks();
if(!joysticks.empty()) { if(!joysticks.empty()) {
// Convert to a C++ bool so that the following calls are resolved correctly even if overloaded.
bool is_pressed = !!isPressed;
switch(key) { switch(key) {
case VK_LeftArrow: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Left, isPressed); break; case VK_LeftArrow: joysticks[0]->set_input(Inputs::Joystick::Input::Left, is_pressed); break;
case VK_RightArrow: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Right, isPressed); break; case VK_RightArrow: joysticks[0]->set_input(Inputs::Joystick::Input::Right, is_pressed); break;
case VK_UpArrow: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Up, isPressed); break; case VK_UpArrow: joysticks[0]->set_input(Inputs::Joystick::Input::Up, is_pressed); break;
case VK_DownArrow: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Down, isPressed); break; case VK_DownArrow: joysticks[0]->set_input(Inputs::Joystick::Input::Down, is_pressed); break;
case VK_Space: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Fire, isPressed); break; case VK_Space: joysticks[0]->set_input(Inputs::Joystick::Input::Fire, is_pressed); break;
case VK_ANSI_A: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput(Inputs::Joystick::DigitalInput::Fire, 0), isPressed); break; case VK_ANSI_A: joysticks[0]->set_input(Inputs::Joystick::Input(Inputs::Joystick::Input::Fire, 0), is_pressed); break;
case VK_ANSI_S: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput(Inputs::Joystick::DigitalInput::Fire, 1), isPressed); break; case VK_ANSI_S: joysticks[0]->set_input(Inputs::Joystick::Input(Inputs::Joystick::Input::Fire, 1), is_pressed); break;
default: default:
if(characters) { if(characters) {
joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput([characters characterAtIndex:0]), isPressed); joysticks[0]->set_input(Inputs::Joystick::Input([characters characterAtIndex:0]), is_pressed);
} else { } else {
joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Fire, isPressed); joysticks[0]->set_input(Inputs::Joystick::Input::Fire, is_pressed);
} }
break; break;
} }

View File

@ -464,16 +464,16 @@ int main(int argc, char *argv[]) {
std::vector<std::unique_ptr<Inputs::Joystick>> &joysticks = joystick_machine->get_joysticks(); std::vector<std::unique_ptr<Inputs::Joystick>> &joysticks = joystick_machine->get_joysticks();
if(!joysticks.empty()) { if(!joysticks.empty()) {
switch(event.key.keysym.scancode) { switch(event.key.keysym.scancode) {
case SDL_SCANCODE_LEFT: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Left, is_pressed); break; case SDL_SCANCODE_LEFT: joysticks[0]->set_input(Inputs::Joystick::Input::Left, is_pressed); break;
case SDL_SCANCODE_RIGHT: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Right, is_pressed); break; case SDL_SCANCODE_RIGHT: joysticks[0]->set_input(Inputs::Joystick::Input::Right, is_pressed); break;
case SDL_SCANCODE_UP: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Up, is_pressed); break; case SDL_SCANCODE_UP: joysticks[0]->set_input(Inputs::Joystick::Input::Up, is_pressed); break;
case SDL_SCANCODE_DOWN: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Down, is_pressed); break; case SDL_SCANCODE_DOWN: joysticks[0]->set_input(Inputs::Joystick::Input::Down, is_pressed); break;
case SDL_SCANCODE_SPACE: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput::Fire, is_pressed); break; case SDL_SCANCODE_SPACE: joysticks[0]->set_input(Inputs::Joystick::Input::Fire, is_pressed); break;
case SDL_SCANCODE_A: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput(Inputs::Joystick::DigitalInput::Fire, 0), is_pressed); break; case SDL_SCANCODE_A: joysticks[0]->set_input(Inputs::Joystick::Input(Inputs::Joystick::Input::Fire, 0), is_pressed); break;
case SDL_SCANCODE_S: joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput(Inputs::Joystick::DigitalInput::Fire, 1), is_pressed); break; case SDL_SCANCODE_S: joysticks[0]->set_input(Inputs::Joystick::Input(Inputs::Joystick::Input::Fire, 1), is_pressed); break;
default: { default: {
const char *key_name = SDL_GetKeyName(event.key.keysym.sym); const char *key_name = SDL_GetKeyName(event.key.keysym.sym);
joysticks[0]->set_digital_input(Inputs::Joystick::DigitalInput(key_name[0]), is_pressed); joysticks[0]->set_input(Inputs::Joystick::Input(key_name[0]), is_pressed);
} break; } break;
} }
} }