JoystickSDL.cc 5.75 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
            newRet[name] = ret[name];
71 72 73
            JoystickSDL *j = (JoystickSDL*)newRet[name];
            if (j->index() != i) {
                j->setIndex(i); // This joystick index has been remapped by SDL
Jacob Walser's avatar
Jacob Walser committed
74 75 76 77
            }
            // 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);
78 79 80
            qCDebug(JoystickLog) << "\tSkipping duplicate" << name;
        }
    }
Jacob Walser's avatar
Jacob Walser committed
81 82 83 84 85 86

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

    ret = newRet;
87 88 89
    return ret;
}

90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
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
105
bool JoystickSDL::_open(void) {
106 107 108 109 110 111
    if ( _isGameController ) {
        sdlController = SDL_GameControllerOpen(_index);
        sdlJoystick = SDL_GameControllerGetJoystick(sdlController);
    } else {
        sdlJoystick = SDL_JoystickOpen(_index);
    }
112 113

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

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

120 121 122
    return true;
}

Gregory Dymarek's avatar
Gregory Dymarek committed
123
void JoystickSDL::_close(void) {
Jacob Walser's avatar
Jacob Walser committed
124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
    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;
144 145
}

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

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

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

169 170 171 172 173 174 175 176
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;
}