JoystickSDL.cc 5.71 KB
Newer Older
1 2 3 4 5
#include "JoystickSDL.h"

#include "QGCApplication.h"

#include <QQmlEngine>
6
#include <QTextStream>
7

8
JoystickSDL::JoystickSDL(const QString& name, int axisCount, int buttonCount, int hatCount, int index, bool isGameController, MultiVehicleManager* multiVehicleManager)
9
    : Joystick(name,axisCount,buttonCount,hatCount,multiVehicleManager)
10
    , _isGameController(isGameController)
11 12
    , _index(index)
{
13
    if(_isGameController) _setDefaultCalibration();
14 15
}

Jacob Walser's avatar
Jacob Walser committed
16
bool JoystickSDL::init(void) {
17
    if (SDL_InitSubSystem(SDL_INIT_GAMECONTROLLER | SDL_INIT_JOYSTICK | SDL_INIT_NOPARACHUTE) < 0) {
Jacob Walser's avatar
Jacob Walser committed
18
        SDL_JoystickEventState(SDL_ENABLE);
19
        qWarning() << "Couldn't initialize SimpleDirectMediaLayer:" << SDL_GetError();
Jacob Walser's avatar
Jacob Walser committed
20
        return false;
21 22
    }

23
    _loadGameControllerMappings();
Jacob Walser's avatar
Jacob Walser committed
24 25 26 27 28 29 30
    return true;
}

QMap<QString, Joystick*> JoystickSDL::discover(MultiVehicleManager* _multiVehicleManager) {
    static QMap<QString, Joystick*> ret;

    QMap<QString,Joystick*> newRet;
31

32 33 34 35 36
    // Load available joysticks

    qCDebug(JoystickLog) << "Available joysticks";

    for (int i=0; i<SDL_NumJoysticks(); i++) {
37 38
        QString name = SDL_JoystickNameForIndex(i);

Jacob Walser's avatar
Jacob Walser committed
39
        // TODO use GUID instead of name in case of two joysticks with same name
40
        if (!ret.contains(name)) {
41
            int axisCount, buttonCount, hatCount;
42
            bool isGameController;
43

44

45 46 47 48 49 50 51
            if (SDL_IsGameController(i)) {
                isGameController = true;
                axisCount = SDL_CONTROLLER_AXIS_MAX;
                buttonCount = SDL_CONTROLLER_BUTTON_MAX;
                hatCount = 0;
            } else {
                isGameController = false;
52 53 54 55 56 57 58 59 60 61 62 63
                if (SDL_Joystick* sdlJoystick = SDL_JoystickOpen(i)) {
                    SDL_ClearError();
                    axisCount = SDL_JoystickNumAxes(sdlJoystick);
                    buttonCount = SDL_JoystickNumButtons(sdlJoystick);
                    hatCount = SDL_JoystickNumHats(sdlJoystick);
                    if (axisCount < 0 || buttonCount < 0 || hatCount < 0) {
                        qCWarning(JoystickLog) << "\t libsdl error parsing joystick features:" << SDL_GetError();
                    }
                    SDL_JoystickClose(sdlJoystick);
                } else {
                    qCWarning(JoystickLog) << "\t libsdl failed opening joystick" << qPrintable(name) << "error:" << SDL_GetError();
                    continue;
64
                }
65 66
            }

67
            qCDebug(JoystickLog) << "\t" << name << "axes:" << axisCount << "buttons:" << buttonCount << "hats:" << hatCount << "isGC:" << isGameController;
Jacob Walser's avatar
Jacob Walser committed
68
            newRet[name] = new JoystickSDL(name, qMax(0,axisCount), qMax(0,buttonCount), qMax(0,hatCount), i, isGameController, _multiVehicleManager);
69
        } else {
Jacob Walser's avatar
Jacob Walser committed
70 71 72 73 74 75 76
            newRet[name] = ret[name];
            if (newRet[name]->index() != i) {
                newRet[name]->setIndex(i); // This joystick index has been remapped by SDL
            }
            // Anything left in ret after we exit the loop has been removed (unplugged) and needs to be cleaned up.
            // We will handle that in JoystickManager in case the removed joystick was in use.
            ret.remove(name);
77 78 79
            qCDebug(JoystickLog) << "\tSkipping duplicate" << name;
        }
    }
Jacob Walser's avatar
Jacob Walser committed
80 81 82 83 84 85

    if (!newRet.count()) {
        qCDebug(JoystickLog) << "\tnone found";
    }

    ret = newRet;
86 87 88
    return ret;
}

89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
void JoystickSDL::_loadGameControllerMappings(void) {
    QFile file(":/db/mapping/joystick/gamecontrollerdb.txt");
    if (!file.open(QIODevice::ReadOnly | QIODevice::Text))
    {
        qWarning() << "Couldn't load GameController mapping database.";
        return;
    }

    QTextStream s(&file);

    while (!s.atEnd()) {
        SDL_GameControllerAddMapping(s.readLine().toStdString().c_str());
    }
}

Gregory Dymarek's avatar
Gregory Dymarek committed
104
bool JoystickSDL::_open(void) {
105 106 107 108 109 110
    if ( _isGameController ) {
        sdlController = SDL_GameControllerOpen(_index);
        sdlJoystick = SDL_GameControllerGetJoystick(sdlController);
    } else {
        sdlJoystick = SDL_JoystickOpen(_index);
    }
111 112

    if (!sdlJoystick) {
Gregory Dymarek's avatar
Gregory Dymarek committed
113 114
        qCWarning(JoystickLog) << "SDL_JoystickOpen failed:" << SDL_GetError();
        return false;
115 116
    }

Jacob Walser's avatar
Jacob Walser committed
117 118
    qCDebug(JoystickLog) << "Opened joystick at" << sdlJoystick;

119 120 121
    return true;
}

Gregory Dymarek's avatar
Gregory Dymarek committed
122
void JoystickSDL::_close(void) {
Jacob Walser's avatar
Jacob Walser committed
123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
    if (sdlJoystick == NULL) {
        qCDebug(JoystickLog) << "Attempt to close null joystick!";
        return;
    }

    qCDebug(JoystickLog) << "Closing" << SDL_JoystickName(sdlJoystick) << "at" << sdlJoystick;

    // We get a segfault if we try to close a joystick that has been detached
    if (SDL_JoystickGetAttached(sdlJoystick) == SDL_FALSE) {
        qCDebug(JoystickLog) << "\tJoystick is not attached!";
    } else {

        if (SDL_JoystickInstanceID(sdlJoystick) != -1) {
            qCDebug(JoystickLog) << "\tID:" << SDL_JoystickInstanceID(sdlJoystick);
            SDL_JoystickClose(sdlJoystick);
        }
    }

    sdlJoystick = NULL;
    sdlController = NULL;
143 144
}

Gregory Dymarek's avatar
Gregory Dymarek committed
145
bool JoystickSDL::_update(void)
146 147
{
    SDL_JoystickUpdate();
148
    SDL_GameControllerUpdate();
149 150 151
    return true;
}

Gregory Dymarek's avatar
Gregory Dymarek committed
152
bool JoystickSDL::_getButton(int i) {
153 154 155 156 157
    if ( _isGameController ) {
        return !!SDL_GameControllerGetButton(sdlController, SDL_GameControllerButton(i));
    } else {
        return !!SDL_JoystickGetButton(sdlJoystick, i);
    }
158 159
}

Gregory Dymarek's avatar
Gregory Dymarek committed
160
int JoystickSDL::_getAxis(int i) {
161 162 163 164 165
    if ( _isGameController ) {
        return SDL_GameControllerGetAxis(sdlController, SDL_GameControllerAxis(i));
    } else {
        return SDL_JoystickGetAxis(sdlJoystick, i);
    }
166 167
}

168 169 170 171 172 173 174 175
uint8_t JoystickSDL::_getHat(int hat,int i) {
    uint8_t hatButtons[] = {SDL_HAT_UP,SDL_HAT_DOWN,SDL_HAT_LEFT,SDL_HAT_RIGHT};

    if ( i < int(sizeof(hatButtons)) ) {
        return !!(SDL_JoystickGetHat(sdlJoystick, hat) & hatButtons[i]);
    }
    return 0;
}