JoystickInput.h 9.81 KB
Newer Older
pixhawk's avatar
pixhawk committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
/*=====================================================================

PIXHAWK Micro Air Vehicle Flying Robotics Toolkit

(c) 2009, 2010 PIXHAWK PROJECT  <http://pixhawk.ethz.ch>

This file is part of the PIXHAWK project

    PIXHAWK is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    PIXHAWK is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with PIXHAWK. If not, see <http://www.gnu.org/licenses/>.

======================================================================*/

/**
 * @file
 *   @brief Definition of joystick interface
 *
 *   @author Lorenz Meier <mavteam@student.ethz.ch>
 *   @author Andreas Romer <mavteam@student.ethz.ch>
 *
 */

#ifndef _JOYSTICKINPUT_H_
#define _JOYSTICKINPUT_H_

#include <QThread>
#include <QList>
38
#include <qmutex.h>
Lorenz Meier's avatar
Lorenz Meier committed
39 40 41
#ifdef Q_OS_MAC
#include <SDL.h>
#else
42
#include <SDL/SDL.h>
Lorenz Meier's avatar
Lorenz Meier committed
43
#endif
pixhawk's avatar
pixhawk committed
44 45 46

#include "UASInterface.h"

47 48 49 50 51 52 53
struct JoystickSettings {
    QMap<int, bool> axesInverted; ///< Whether each axis should be used inverted from what was reported.
    QMap<int, bool> axesLimited; ///< Whether each axis should be limited to only the positive range. Currently this only applies to the throttle axis, but is kept generic here to possibly support other axes.
    QMap<int, int> buttonActions; ///< The index of the action associated with every button.
};
Q_DECLARE_METATYPE(JoystickSettings);

pixhawk's avatar
pixhawk committed
54 55 56 57 58 59 60 61 62
/**
 * @brief Joystick input
 */
class JoystickInput : public QThread
{
    Q_OBJECT

public:
    JoystickInput();
63
    ~JoystickInput();
pixhawk's avatar
pixhawk committed
64
    void run();
65
    void shutdown();
pixhawk's avatar
pixhawk committed
66

67 68 69 70 71 72 73 74 75 76 77 78 79
    /**
     * @brief The JOYSTICK_INPUT_MAPPING enum storing the values for each item in the mapping combobox.
     * This should match the order of items in the mapping combobox in JoystickAxis.ui.
     */
    enum JOYSTICK_INPUT_MAPPING
    {
        JOYSTICK_INPUT_MAPPING_NONE     = 0,
        JOYSTICK_INPUT_MAPPING_YAW      = 1,
        JOYSTICK_INPUT_MAPPING_PITCH    = 2,
        JOYSTICK_INPUT_MAPPING_ROLL     = 3,
        JOYSTICK_INPUT_MAPPING_THROTTLE = 4
    };

Lorenz Meier's avatar
Lorenz Meier committed
80
    /**
81
     * @brief Load joystick-specific settings.
Lorenz Meier's avatar
Lorenz Meier committed
82
     */
83 84 85 86 87
    void loadJoystickSettings();
    /**
     * @brief Load joystick-independent settings.
     */
    void loadGeneralSettings();
Lorenz Meier's avatar
Lorenz Meier committed
88 89

    /**
90 91 92 93 94
     * @brief Store joystick-specific settings.
     */
    void storeJoystickSettings() const;
    /**
     * @brief Store joystick-independent settings.
Lorenz Meier's avatar
Lorenz Meier committed
95
     */
96 97 98 99 100 101
    void storeGeneralSettings() const;

    bool enabled() const
    {
        return isEnabled;
    }
Lorenz Meier's avatar
Lorenz Meier committed
102

103
    int getMappingThrottleAxis() const
Lorenz Meier's avatar
Lorenz Meier committed
104
    {
105
        return throttleAxis;
Lorenz Meier's avatar
Lorenz Meier committed
106 107
    }

108
    int getMappingRollAxis() const
Lorenz Meier's avatar
Lorenz Meier committed
109
    {
110
        return rollAxis;
Lorenz Meier's avatar
Lorenz Meier committed
111 112
    }

113
    int getMappingPitchAxis() const
Lorenz Meier's avatar
Lorenz Meier committed
114
    {
115
        return pitchAxis;
Lorenz Meier's avatar
Lorenz Meier committed
116 117
    }

118
    int getMappingYawAxis() const
Lorenz Meier's avatar
Lorenz Meier committed
119 120 121 122
    {
        return yawAxis;
    }

123 124
    int getJoystickNumButtons() const
    {
125
        return joystickNumButtons;
126 127
    }

128 129
    int getJoystickNumAxes() const
    {
130
        return joystickNumAxes;
131 132
    }

133 134 135 136 137
    int getJoystickID() const
    {
        return joystickID;
    }

138 139 140 141 142
    const QString& getName() const
    {
        return joystickName;
    }

143 144
    int getNumJoysticks() const
    {
145
        return numJoysticks;
146 147 148 149 150 151 152
    }

    QString getJoystickNameById(int id) const
    {
        return QString(SDL_JoystickName(id));
    }

153 154 155 156
    float getCurrentValueForAxis(int axis) const;
    bool getInvertedForAxis(int axis) const;
    bool getRangeLimitForAxis(int axis) const;
    int getActionForButton(int button) const;
157

pixhawk's avatar
pixhawk committed
158 159 160 161 162 163
    const double sdlJoystickMin;
    const double sdlJoystickMax;

protected:
    double calibrationPositive[10];
    double calibrationNegative[10];
164 165 166 167

    bool isEnabled; ///< Track whether the system should emit the higher-level signals: joystickChanged & actionTriggered.
    bool done;

pixhawk's avatar
pixhawk committed
168
    SDL_Joystick* joystick;
169
    UASInterface* uas; ///< Track the current UAS.
170 171
    int autopilotType; ///< Cache the autopilotType
    int systemType; ///< Cache the systemType
172
    bool uasCanReverse; ///< Track whether the connect UAS can drive a reverse speed.
pixhawk's avatar
pixhawk committed
173

174
    // Store the mapping between axis numbers and the roll/pitch/yaw/throttle configuration.
175
    // Value is one of JoystickAxis::JOYSTICK_INPUT_MAPPING.
176 177
    int rollAxis;
    int pitchAxis;
pixhawk's avatar
pixhawk committed
178
    int yawAxis;
179 180 181
    int throttleAxis;

    // Cache information on the joystick instead of polling the SDL everytime.
182
    int numJoysticks; ///< Total number of joysticks detected by the SDL.
183
    QString joystickName;
184
    int joystickID;
185 186
    int joystickNumAxes;
    int joystickNumButtons;
187

188 189 190 191 192 193 194 195 196
    // Track axis/button settings based on a Joystick/AutopilotType/SystemType triplet.
    // This is only a double-map, because settings are stored/loaded based on joystick
    // name first, so only the settings for the current joystick need to be stored at any given time.
    // Pointers are kept to the various settings field to reduce lookup times.
    // Note that the mapping (0,0) corresponds to when no UAS is connected. Since this corresponds
    // to a generic vehicle type and a generic autopilot, this is a pretty safe default.
    QMap<int, QMap<int, JoystickSettings> > joystickSettings;

    // Track the last state of the axes, buttons, and hats for only emitting change signals.
197
    QList<float> joystickAxes; ///< The values of every axes during the last sample.
198 199
    quint16 joystickButtons;   ///< The state of every button. Bitfield supporting 16 buttons with 1s indicating that the button is down.
    int xHat, yHat;            ///< The horizontal/vertical hat directions. Values are -1, 0, 1, with (-1,-1) indicating bottom-left.
pixhawk's avatar
pixhawk committed
200 201 202

    void init();

203
signals:
pixhawk's avatar
pixhawk committed
204 205 206 207

    /**
     * @brief Signal containing all joystick raw positions
     *
208 209 210 211
     * @param roll forward / pitch / x axis, front: 1, center: 0, back: -1. If the roll axis isn't defined, NaN is transmit instead.
     * @param pitch left / roll / y axis, left: -1, middle: 0, right: 1. If the roll axis isn't defined, NaN is transmit instead.
     * @param yaw turn axis, left-turn: -1, centered: 0, right-turn: 1. If the roll axis isn't defined, NaN is transmit instead.
     * @param throttle Throttle, -100%:-1.0, 0%: 0.0, 100%: 1.0. If the roll axis isn't defined, NaN is transmit instead.
pixhawk's avatar
pixhawk committed
212 213 214
     * @param xHat hat vector in forward-backward direction, +1 forward, 0 center, -1 backward
     * @param yHat hat vector in left-right direction, -1 left, 0 center, +1 right
     */
215
    void joystickChanged(double roll, double pitch, double yaw, double throttle, int xHat, int yHat, int buttons);
pixhawk's avatar
pixhawk committed
216 217

    /**
218
      * @brief Emit a new value for an axis
219
      *
220
      * @param value Value of the axis, between -1.0 and 1.0.
221
      */
222
    void axisValueChanged(int axis, float value);
pixhawk's avatar
pixhawk committed
223

224
    /**
225
      * @brief Joystick button has changed state from unpressed to pressed.
226 227
      * @param key index of the pressed key
      */
pixhawk's avatar
pixhawk committed
228 229
    void buttonPressed(int key);

230 231 232 233 234 235 236
    /**
      * @brief Joystick button has changed state from pressed to unpressed.
      *
      * @param key index of the released key
      */
    void buttonReleased(int key);

237 238 239 240 241 242
    /**
      * @brief A joystick button was pressed that had a corresponding action.
      * @param action The index of the action to trigger. Dependent on UAS.
      */
    void actionTriggered(int action);

243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
    /**
      * @brief Hat (8-way switch on the top) has changed position
      *
      * Coordinate frame for joystick hat:
      *
      *    y
      *    ^
      *    |
      *    |
      *    0 ----> x
      *
      *
      * @param x vector in left-right direction
      * @param y vector in forward-backward direction
      */
pixhawk's avatar
pixhawk committed
258 259
    void hatDirectionChanged(int x, int y);

260 261 262
    /** @brief Signals that new joystick-specific settings were changed. Useful for triggering updates that at dependent on the current joystick. */
    void joystickSettingsChanged();

263
public slots:
264 265
    /** @brief Enable or disable emitting the high-level control signals from the joystick. */
    void setEnabled(bool enable);
266
    /** @brief Specify the UAS that this input should forward joystickChanged signals and buttonPresses to. */
pixhawk's avatar
pixhawk committed
267
    void setActiveUAS(UASInterface* uas);
268
    /** @brief Switch to a new joystick by ID number. Both buttons and axes are updated with the proper signals emitted. */
269
    void setActiveJoystick(int id);
270 271 272 273 274 275
    /**
     * @brief Change the control mapping for a given joystick axis.
     * @param axisID The axis to modify (0-indexed)
     * @param newMapping The mapping to use.
     * @see JOYSTICK_INPUT_MAPPING
     */
276
    void setAxisMapping(int axis, JoystickInput::JOYSTICK_INPUT_MAPPING newMapping);
277 278 279 280 281 282
    /**
     * @brief Specify if an axis should be inverted.
     * @param axis The ID of the axis.
     * @param inverted True indicates inverted from normal. Varies by controller.
     */
    void setAxisInversion(int axis, bool inverted);
283 284 285 286 287 288
    /**
     * @brief Specify that an axis should only transmit the positive values. Useful for controlling throttle from auto-centering axes.
     * @param axis Which axis has its range limited.
     * @param limitRange If true only the positive half of this axis will be read.
     */
    void setAxisRangeLimit(int axis, bool limitRange);
289 290 291 292 293 294 295 296
    /**
     * @brief Specify a button->action mapping for the given uas.
     * This mapping is applied based on UAS autopilot type and UAS system type.
     * Connects the buttonEmitted signal for the corresponding button to the corresponding action for the current UAS.
     * @param button The numeric ID for the button
     * @param action The numeric ID of the action for this UAS to map to.
     */
    void setButtonAction(int button, int action);
pixhawk's avatar
pixhawk committed
297 298 299
};

#endif // _JOYSTICKINPUT_H_