//============================================================================ // // SSSS tt lll lll // SS SS tt ll ll // SS tttttt eeee ll ll aaaa // SSSS tt ee ee ll ll aa // SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator" // SS SS tt ee ll ll aa aa // SSSS ttt eeeee llll llll aaaaa // // Copyright (c) 1995-2019 by Bradford W. Mott, Stephen Anthony // and the Stella Team // // See the file "License.txt" for information on usage and redistribution of // this file, and for a DISCLAIMER OF ALL WARRANTIES. //============================================================================ #ifndef PADDLES_HXX #define PADDLES_HXX #include "bspf.hxx" #include "Control.hxx" #include "Event.hxx" /** The standard Atari 2600 pair of paddle controllers. @author Bradford W. Mott */ class Paddles : public Controller { public: /** Create a new pair of paddle controllers plugged into the specified jack @param jack The jack the controller is plugged into @param event The event object to use for events @param system The system using this controller @param swappaddle Whether to swap the paddles plugged into this jack @param swapaxis Whether to swap the axis on the paddle (x <-> y) @param swapdir Whether to swap the direction for which an axis causes movement (lesser axis values cause paddle resistance to decrease instead of increase) */ Paddles(Jack jack, const Event& event, const System& system, bool swappaddle, bool swapaxis, bool swapdir); virtual ~Paddles() = default; public: static constexpr int MAX_DIGITAL_SENSE = 20; static constexpr int MAX_MOUSE_SENSE = 20; static constexpr int MIN_DEJITTER = 0; static constexpr int MAX_DEJITTER = 10; /** Update the entire digital and analog pin state according to the events currently set. */ void update() override; /** Returns the name of this controller. */ string name() const override { return "Paddles"; } /** Answers whether the controller is intrinsically an analog controller. */ bool isAnalog() const override { return true; } /** Determines how this controller will treat values received from the X/Y axis and left/right buttons of the mouse. Since not all controllers use the mouse the same way (or at all), it's up to the specific class to decide how to use this data. In the current implementation, the left button is tied to the X axis, and the right one tied to the Y axis. @param xtype The controller to use for x-axis data @param xid The controller ID to use for x-axis data (-1 for no id) @param ytype The controller to use for y-axis data @param yid The controller ID to use for y-axis data (-1 for no id) @return Whether the controller supports using the mouse */ bool setMouseControl(Controller::Type xtype, int xid, Controller::Type ytype, int yid) override; /** @param strength Value from 0 to 10 */ static void setDejitterBase(int strength); /** @param strength Value from 0 to 10 */ static void setDejitterDiff(int strength); /** Sets the sensitivity for digital emulation of paddle movement. This is only used for *digital* events (ie, buttons or keys, or digital joystick axis events); Stelladaptors or the mouse are not modified. @param sensitivity Value from 1 to MAX_DIGITAL_SENSE, with larger values causing more movement */ static void setDigitalSensitivity(int sensitivity); /** Sets the sensitivity for analog emulation of paddle movement using a mouse. @param sensitivity Value from 1 to MAX_MOUSE_SENSE, with larger values causing more movement */ static void setMouseSensitivity(int sensitivity); /** Sets the maximum upper range for digital/mouse emulation of paddle movement (ie, a value of 50 means to only use 50% of the possible range of movement). Note that this specfically does not apply to Stelladaptor-like devices, which uses an absolute value range. @param range Value from 1 to 100, representing the percentage of the range to use */ static void setPaddleRange(int range); static constexpr double MAX_RESISTANCE = 1400000.0; private: // Range of values over which digital and mouse movement is scaled // to paddle resistance static constexpr int TRIGMIN = 1; static constexpr int TRIGMAX = 4096; static int TRIGRANGE; // This one is variable for the upper range // Pre-compute the events we care about based on given port // This will eliminate test for left or right port in update() Event::Type myP0AxisValue, myP1AxisValue, myP0DecEvent, myP0IncEvent, myP1DecEvent, myP1IncEvent, myP0FireEvent, myP1FireEvent, myAxisMouseMotion; // The following are used for the various mouse-axis modes int myMPaddleID{-1}; // paddle to emulate in 'automatic' mode int myMPaddleIDX{-1}, myMPaddleIDY{-1}; // paddles to emulate in 'specific axis' mode bool myKeyRepeat0{false}, myKeyRepeat1{false}; int myPaddleRepeat0{0}, myPaddleRepeat1{0}; std::array myCharge{TRIGRANGE / 2}, myLastCharge{0}; int myLastAxisX{0}, myLastAxisY{0}; int myAxisDigitalZero{0}, myAxisDigitalOne{0}; static int DIGITAL_SENSITIVITY, DIGITAL_DISTANCE; static int DEJITTER_BASE, DEJITTER_DIFF; static int MOUSE_SENSITIVITY; // Lookup table for associating paddle buttons with controller pins // Yes, this is hideously complex static const std::array ourButtonPin; private: // Following constructors and assignment operators not supported Paddles() = delete; Paddles(const Paddles&) = delete; Paddles(Paddles&&) = delete; Paddles& operator=(const Paddles&) = delete; Paddles& operator=(Paddles&&) = delete; }; #endif