/* Copyright (c) 2013-2014 Jeffrey Pfau * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "InputController.h" #include "ConfigController.h" #include "GamepadAxisEvent.h" #include "GamepadButtonEvent.h" #include "InputProfile.h" #include "LogController.h" #include #include #include #ifdef BUILD_QT_MULTIMEDIA #include #include #endif #include #include using namespace QGBA; #ifdef BUILD_SDL int InputController::s_sdlInited = 0; mSDLEvents InputController::s_sdlEvents; #endif InputController::InputController(int playerId, QWidget* topLevel, QObject* parent) : QObject(parent) , m_playerId(playerId) , m_topLevel(topLevel) , m_focusParent(topLevel) { mInputMapInit(&m_inputMap, &GBAInputInfo); #ifdef BUILD_SDL if (s_sdlInited == 0) { mSDLInitEvents(&s_sdlEvents); } ++s_sdlInited; m_sdlPlayer.bindings = &m_inputMap; mSDLInitBindingsGBA(&m_inputMap); updateJoysticks(); #endif #ifdef BUILD_SDL connect(&m_gamepadTimer, &QTimer::timeout, [this]() { testGamepad(SDL_BINDING_BUTTON); if (m_playerId == 0) { updateJoysticks(); } }); #endif m_gamepadTimer.setInterval(50); m_gamepadTimer.start(); #ifdef BUILD_QT_MULTIMEDIA connect(&m_videoDumper, &VideoDumper::imageAvailable, this, &InputController::setCamImage); #endif mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_X, GBA_KEY_A); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Z, GBA_KEY_B); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_A, GBA_KEY_L); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_S, GBA_KEY_R); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Return, GBA_KEY_START); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Backspace, GBA_KEY_SELECT); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Up, GBA_KEY_UP); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Down, GBA_KEY_DOWN); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Left, GBA_KEY_LEFT); mInputBindKey(&m_inputMap, KEYBOARD, Qt::Key_Right, GBA_KEY_RIGHT); #ifdef M_CORE_GBA m_lux.p = this; m_lux.sample = [](GBALuminanceSource* context) { InputControllerLux* lux = static_cast(context); lux->value = 0xFF - lux->p->m_luxValue; }; m_lux.readLuminance = [](GBALuminanceSource* context) { InputControllerLux* lux = static_cast(context); return lux->value; }; setLuminanceLevel(0); #endif m_image.p = this; m_image.startRequestImage = [](mImageSource* context, unsigned w, unsigned h, int) { InputControllerImage* image = static_cast(context); image->w = w; image->h = h; if (image->image.isNull()) { image->image.load(":/res/no-cam.png"); } #ifdef BUILD_QT_MULTIMEDIA if (image->p->m_config->getQtOption("cameraDriver").toInt() == static_cast(CameraDriver::QT_MULTIMEDIA)) { QByteArray camera = image->p->m_config->getQtOption("camera").toByteArray(); if (!camera.isNull()) { QMetaObject::invokeMethod(image->p, "setCamera", Q_ARG(QByteArray, camera)); } QMetaObject::invokeMethod(image->p, "setupCam"); } #endif }; m_image.stopRequestImage = [](mImageSource* context) { InputControllerImage* image = static_cast(context); #ifdef BUILD_QT_MULTIMEDIA QMetaObject::invokeMethod(image->p, "teardownCam"); #endif }; m_image.requestImage = [](mImageSource* context, const void** buffer, size_t* stride, mColorFormat* format) { InputControllerImage* image = static_cast(context); QSize size; { QMutexLocker locker(&image->mutex); if (image->outOfDate) { image->resizedImage = image->image.scaled(image->w, image->h, Qt::KeepAspectRatioByExpanding); image->resizedImage = image->resizedImage.convertToFormat(QImage::Format_RGB16); image->outOfDate = false; } } size = image->resizedImage.size(); const uint16_t* bits = reinterpret_cast(image->resizedImage.constBits()); if (size.width() > image->w) { bits += (size.width() - image->w) / 2; } if (size.height() > image->h) { bits += ((size.height() - image->h) / 2) * size.width(); } *buffer = bits; *stride = image->resizedImage.bytesPerLine() / sizeof(*bits); *format = mCOLOR_RGB565; }; } InputController::~InputController() { mInputMapDeinit(&m_inputMap); #ifdef BUILD_SDL if (m_playerAttached) { mSDLDetachPlayer(&s_sdlEvents, &m_sdlPlayer); } --s_sdlInited; if (s_sdlInited == 0) { mSDLDeinitEvents(&s_sdlEvents); } #endif } void InputController::setConfiguration(ConfigController* config) { m_config = config; loadConfiguration(KEYBOARD); #ifdef BUILD_SDL mSDLEventsLoadConfig(&s_sdlEvents, config->input()); if (!m_playerAttached) { m_playerAttached = mSDLAttachPlayer(&s_sdlEvents, &m_sdlPlayer); } loadConfiguration(SDL_BINDING_BUTTON); loadProfile(SDL_BINDING_BUTTON, profileForType(SDL_BINDING_BUTTON)); #endif } void InputController::loadConfiguration(uint32_t type) { mInputMapLoad(&m_inputMap, type, m_config->input()); #ifdef BUILD_SDL if (m_playerAttached) { mSDLPlayerLoadConfig(&m_sdlPlayer, m_config->input()); } #endif } void InputController::loadProfile(uint32_t type, const QString& profile) { bool loaded = mInputProfileLoad(&m_inputMap, type, m_config->input(), profile.toUtf8().constData()); recalibrateAxes(); if (!loaded) { const InputProfile* ip = InputProfile::findProfile(profile); if (ip) { ip->apply(this); } } emit profileLoaded(profile); } void InputController::saveConfiguration() { saveConfiguration(KEYBOARD); #ifdef BUILD_SDL saveConfiguration(SDL_BINDING_BUTTON); saveProfile(SDL_BINDING_BUTTON, profileForType(SDL_BINDING_BUTTON)); if (m_playerAttached) { mSDLPlayerSaveConfig(&m_sdlPlayer, m_config->input()); } #endif m_config->write(); } void InputController::saveConfiguration(uint32_t type) { mInputMapSave(&m_inputMap, type, m_config->input()); m_config->write(); } void InputController::saveProfile(uint32_t type, const QString& profile) { mInputProfileSave(&m_inputMap, type, m_config->input(), profile.toUtf8().constData()); m_config->write(); } const char* InputController::profileForType(uint32_t type) { UNUSED(type); #ifdef BUILD_SDL if (type == SDL_BINDING_BUTTON && m_sdlPlayer.joystick) { #if SDL_VERSION_ATLEAST(2, 0, 0) return SDL_JoystickName(m_sdlPlayer.joystick->joystick); #else return SDL_JoystickName(SDL_JoystickIndex(m_sdlPlayer.joystick->joystick)); #endif } #endif return 0; } QStringList InputController::connectedGamepads(uint32_t type) const { UNUSED(type); #ifdef BUILD_SDL if (type == SDL_BINDING_BUTTON) { QStringList pads; for (size_t i = 0; i < SDL_JoystickListSize(&s_sdlEvents.joysticks); ++i) { const char* name; #if SDL_VERSION_ATLEAST(2, 0, 0) name = SDL_JoystickName(SDL_JoystickListGetPointer(&s_sdlEvents.joysticks, i)->joystick); #else name = SDL_JoystickName(SDL_JoystickIndex(SDL_JoystickListGetPointer(&s_sdlEvents.joysticks, i)->joystick)); #endif if (name) { pads.append(QString(name)); } else { pads.append(QString()); } } return pads; } #endif return QStringList(); } int InputController::gamepad(uint32_t type) const { #ifdef BUILD_SDL if (type == SDL_BINDING_BUTTON) { return m_sdlPlayer.joystick ? m_sdlPlayer.joystick->index : 0; } #endif return 0; } void InputController::setGamepad(uint32_t type, int index) { #ifdef BUILD_SDL if (type == SDL_BINDING_BUTTON) { mSDLPlayerChangeJoystick(&s_sdlEvents, &m_sdlPlayer, index); } #endif } void InputController::setPreferredGamepad(uint32_t type, int index) { if (!m_config) { return; } #ifdef BUILD_SDL #if SDL_VERSION_ATLEAST(2, 0, 0) char name[34] = {0}; SDL_JoystickGetGUIDString(SDL_JoystickGetGUID(SDL_JoystickListGetPointer(&s_sdlEvents.joysticks, index)->joystick), name, sizeof(name)); #else const char* name = SDL_JoystickName(SDL_JoystickIndex(SDL_JoystickListGetPointer(&s_sdlEvents.joysticks, index)->joystick)); if (!name) { return; } #endif mInputSetPreferredDevice(m_config->input(), "gba", type, m_playerId, name); #else UNUSED(type); UNUSED(index); #endif } mRumble* InputController::rumble() { #ifdef BUILD_SDL #if SDL_VERSION_ATLEAST(2, 0, 0) if (m_playerAttached) { return &m_sdlPlayer.rumble.d; } #endif #endif return nullptr; } mRotationSource* InputController::rotationSource() { #ifdef BUILD_SDL if (m_playerAttached) { return &m_sdlPlayer.rotation.d; } #endif return nullptr; } void InputController::registerTiltAxisX(int axis) { #ifdef BUILD_SDL if (m_playerAttached) { m_sdlPlayer.rotation.axisX = axis; } #endif } void InputController::registerTiltAxisY(int axis) { #ifdef BUILD_SDL if (m_playerAttached) { m_sdlPlayer.rotation.axisY = axis; } #endif } void InputController::registerGyroAxisX(int axis) { #ifdef BUILD_SDL if (m_playerAttached) { m_sdlPlayer.rotation.gyroX = axis; } #endif } void InputController::registerGyroAxisY(int axis) { #ifdef BUILD_SDL if (m_playerAttached) { m_sdlPlayer.rotation.gyroY = axis; } #endif } float InputController::gyroSensitivity() const { #ifdef BUILD_SDL if (m_playerAttached) { return m_sdlPlayer.rotation.gyroSensitivity; } #endif return 0; } void InputController::setGyroSensitivity(float sensitivity) { #ifdef BUILD_SDL if (m_playerAttached) { m_sdlPlayer.rotation.gyroSensitivity = sensitivity; } #endif } GBAKey InputController::mapKeyboard(int key) const { return static_cast(mInputMapKey(&m_inputMap, KEYBOARD, key)); } void InputController::bindKey(uint32_t type, int key, GBAKey gbaKey) { return mInputBindKey(&m_inputMap, type, key, gbaKey); } void InputController::updateJoysticks() { #ifdef BUILD_SDL QString profile = profileForType(SDL_BINDING_BUTTON); mSDLUpdateJoysticks(&s_sdlEvents, m_config->input()); QString newProfile = profileForType(SDL_BINDING_BUTTON); if (profile != newProfile) { loadProfile(SDL_BINDING_BUTTON, newProfile); } #endif } int InputController::pollEvents() { int activeButtons = 0; #ifdef BUILD_SDL if (m_playerAttached && m_sdlPlayer.joystick) { SDL_Joystick* joystick = m_sdlPlayer.joystick->joystick; SDL_JoystickUpdate(); int numButtons = SDL_JoystickNumButtons(joystick); int i; for (i = 0; i < numButtons; ++i) { GBAKey key = static_cast(mInputMapKey(&m_inputMap, SDL_BINDING_BUTTON, i)); if (key == GBA_KEY_NONE) { continue; } if (hasPendingEvent(key)) { continue; } if (SDL_JoystickGetButton(joystick, i)) { activeButtons |= 1 << key; } } int numHats = SDL_JoystickNumHats(joystick); for (i = 0; i < numHats; ++i) { int hat = SDL_JoystickGetHat(joystick, i); activeButtons |= mInputMapHat(&m_inputMap, SDL_BINDING_BUTTON, i, hat); } int numAxes = SDL_JoystickNumAxes(joystick); for (i = 0; i < numAxes; ++i) { int value = SDL_JoystickGetAxis(joystick, i); enum GBAKey key = static_cast(mInputMapAxis(&m_inputMap, SDL_BINDING_BUTTON, i, value)); if (key != GBA_KEY_NONE) { activeButtons |= 1 << key; } } } #endif return activeButtons; } QSet InputController::activeGamepadButtons(int type) { QSet activeButtons; #ifdef BUILD_SDL if (m_playerAttached && type == SDL_BINDING_BUTTON && m_sdlPlayer.joystick) { SDL_Joystick* joystick = m_sdlPlayer.joystick->joystick; SDL_JoystickUpdate(); int numButtons = SDL_JoystickNumButtons(joystick); int i; for (i = 0; i < numButtons; ++i) { if (SDL_JoystickGetButton(joystick, i)) { activeButtons.insert(i); } } } #endif return activeButtons; } void InputController::recalibrateAxes() { #ifdef BUILD_SDL if (m_playerAttached && m_sdlPlayer.joystick) { SDL_Joystick* joystick = m_sdlPlayer.joystick->joystick; SDL_JoystickUpdate(); int numAxes = SDL_JoystickNumAxes(joystick); if (numAxes < 1) { return; } m_deadzones.resize(numAxes); int i; for (i = 0; i < numAxes; ++i) { m_deadzones[i] = SDL_JoystickGetAxis(joystick, i); } } #endif } QSet> InputController::activeGamepadAxes(int type) { QSet> activeAxes; #ifdef BUILD_SDL if (m_playerAttached && type == SDL_BINDING_BUTTON && m_sdlPlayer.joystick) { SDL_Joystick* joystick = m_sdlPlayer.joystick->joystick; SDL_JoystickUpdate(); int numAxes = SDL_JoystickNumAxes(joystick); if (numAxes < 1) { return activeAxes; } m_deadzones.resize(numAxes); int i; for (i = 0; i < numAxes; ++i) { int32_t axis = SDL_JoystickGetAxis(joystick, i); axis -= m_deadzones[i]; if (axis >= AXIS_THRESHOLD || axis <= -AXIS_THRESHOLD) { activeAxes.insert(qMakePair(i, axis > 0 ? GamepadAxisEvent::POSITIVE : GamepadAxisEvent::NEGATIVE)); } } } #endif return activeAxes; } void InputController::bindAxis(uint32_t type, int axis, GamepadAxisEvent::Direction direction, GBAKey key) { const mInputAxis* old = mInputQueryAxis(&m_inputMap, type, axis); mInputAxis description = { GBA_KEY_NONE, GBA_KEY_NONE, -AXIS_THRESHOLD, AXIS_THRESHOLD }; if (old) { description = *old; } int deadzone = 0; if (axis > 0 && m_deadzones.size() > axis) { deadzone = m_deadzones[axis]; } switch (direction) { case GamepadAxisEvent::NEGATIVE: description.lowDirection = key; description.deadLow = deadzone - AXIS_THRESHOLD; break; case GamepadAxisEvent::POSITIVE: description.highDirection = key; description.deadHigh = deadzone + AXIS_THRESHOLD; break; default: return; } mInputBindAxis(&m_inputMap, type, axis, &description); } void InputController::unbindAllAxes(uint32_t type) { mInputUnbindAllAxes(&m_inputMap, type); } QSet> InputController::activeGamepadHats(int type) { QSet> activeHats; #ifdef BUILD_SDL if (m_playerAttached && type == SDL_BINDING_BUTTON && m_sdlPlayer.joystick) { SDL_Joystick* joystick = m_sdlPlayer.joystick->joystick; SDL_JoystickUpdate(); int numHats = SDL_JoystickNumHats(joystick); if (numHats < 1) { return activeHats; } int i; for (i = 0; i < numHats; ++i) { int hat = SDL_JoystickGetHat(joystick, i); if (hat & GamepadHatEvent::UP) { activeHats.insert(qMakePair(i, GamepadHatEvent::UP)); } if (hat & GamepadHatEvent::RIGHT) { activeHats.insert(qMakePair(i, GamepadHatEvent::RIGHT)); } if (hat & GamepadHatEvent::DOWN) { activeHats.insert(qMakePair(i, GamepadHatEvent::DOWN)); } if (hat & GamepadHatEvent::LEFT) { activeHats.insert(qMakePair(i, GamepadHatEvent::LEFT)); } } } #endif return activeHats; } void InputController::bindHat(uint32_t type, int hat, GamepadHatEvent::Direction direction, GBAKey gbaKey) { mInputHatBindings bindings{ -1, -1, -1, -1 }; mInputQueryHat(&m_inputMap, type, hat, &bindings); switch (direction) { case GamepadHatEvent::UP: bindings.up = gbaKey; break; case GamepadHatEvent::RIGHT: bindings.right = gbaKey; break; case GamepadHatEvent::DOWN: bindings.down = gbaKey; break; case GamepadHatEvent::LEFT: bindings.left = gbaKey; break; default: return; } mInputBindHat(&m_inputMap, type, hat, &bindings); } void InputController::testGamepad(int type) { auto activeAxes = activeGamepadAxes(type); auto oldAxes = m_activeAxes; m_activeAxes = activeAxes; auto activeButtons = activeGamepadButtons(type); auto oldButtons = m_activeButtons; m_activeButtons = activeButtons; auto activeHats = activeGamepadHats(type); auto oldHats = m_activeHats; m_activeHats = activeHats; if (!QApplication::focusWidget()) { return; } activeAxes.subtract(oldAxes); oldAxes.subtract(m_activeAxes); for (auto& axis : m_activeAxes) { bool newlyAboveThreshold = activeAxes.contains(axis); if (newlyAboveThreshold) { GamepadAxisEvent* event = new GamepadAxisEvent(axis.first, axis.second, newlyAboveThreshold, type, this); postPendingEvent(event->gbaKey()); sendGamepadEvent(event); if (!event->isAccepted()) { clearPendingEvent(event->gbaKey()); } } } for (auto axis : oldAxes) { GamepadAxisEvent* event = new GamepadAxisEvent(axis.first, axis.second, false, type, this); clearPendingEvent(event->gbaKey()); sendGamepadEvent(event); } if (!QApplication::focusWidget()) { return; } activeButtons.subtract(oldButtons); oldButtons.subtract(m_activeButtons); for (int button : activeButtons) { GamepadButtonEvent* event = new GamepadButtonEvent(GamepadButtonEvent::Down(), button, type, this); postPendingEvent(event->gbaKey()); sendGamepadEvent(event); if (!event->isAccepted()) { clearPendingEvent(event->gbaKey()); } } for (int button : oldButtons) { GamepadButtonEvent* event = new GamepadButtonEvent(GamepadButtonEvent::Up(), button, type, this); clearPendingEvent(event->gbaKey()); sendGamepadEvent(event); } activeHats.subtract(oldHats); oldHats.subtract(m_activeHats); for (auto& hat : activeHats) { GamepadHatEvent* event = new GamepadHatEvent(GamepadHatEvent::Down(), hat.first, hat.second, type, this); postPendingEvent(event->gbaKey()); sendGamepadEvent(event); if (!event->isAccepted()) { clearPendingEvent(event->gbaKey()); } } for (auto& hat : oldHats) { GamepadHatEvent* event = new GamepadHatEvent(GamepadHatEvent::Up(), hat.first, hat.second, type, this); clearPendingEvent(event->gbaKey()); sendGamepadEvent(event); } } void InputController::sendGamepadEvent(QEvent* event) { QWidget* focusWidget = nullptr; if (m_focusParent) { focusWidget = m_focusParent->focusWidget(); if (!focusWidget) { focusWidget = m_focusParent; } } else { focusWidget = QApplication::focusWidget(); } QApplication::sendEvent(focusWidget, event); } void InputController::postPendingEvent(GBAKey key) { m_pendingEvents.insert(key); } void InputController::clearPendingEvent(GBAKey key) { m_pendingEvents.remove(key); } bool InputController::hasPendingEvent(GBAKey key) const { return m_pendingEvents.contains(key); } void InputController::suspendScreensaver() { #ifdef BUILD_SDL #if SDL_VERSION_ATLEAST(2, 0, 0) mSDLSuspendScreensaver(&s_sdlEvents); #endif #endif } void InputController::resumeScreensaver() { #ifdef BUILD_SDL #if SDL_VERSION_ATLEAST(2, 0, 0) mSDLResumeScreensaver(&s_sdlEvents); #endif #endif } void InputController::setScreensaverSuspendable(bool suspendable) { #ifdef BUILD_SDL #if SDL_VERSION_ATLEAST(2, 0, 0) mSDLSetScreensaverSuspendable(&s_sdlEvents, suspendable); #endif #endif } void InputController::stealFocus(QWidget* focus) { m_focusParent = focus; } void InputController::releaseFocus(QWidget* focus) { if (focus == m_focusParent) { m_focusParent = m_topLevel; } } void InputController::loadCamImage(const QString& path) { setCamImage(QImage(path)); } void InputController::setCamImage(const QImage& image) { if (image.isNull()) { return; } QMutexLocker locker(&m_image.mutex); m_image.image = image; m_image.resizedImage = QImage(); m_image.outOfDate = true; } QList> InputController::listCameras() const { QList> out; #ifdef BUILD_QT_MULTIMEDIA QList cams = QCameraInfo::availableCameras(); for (const auto& cam : cams) { out.append(qMakePair(cam.deviceName().toLatin1(), cam.description())); } #endif return out; } void InputController::increaseLuminanceLevel() { setLuminanceLevel(m_luxLevel + 1); } void InputController::decreaseLuminanceLevel() { setLuminanceLevel(m_luxLevel - 1); } void InputController::setLuminanceLevel(int level) { int value = 0x16; level = std::max(0, std::min(10, level)); if (level > 0) { value += GBA_LUX_LEVELS[level - 1]; } setLuminanceValue(value); } void InputController::setLuminanceValue(uint8_t value) { m_luxValue = value; value = std::max(value - 0x16, 0); m_luxLevel = 10; for (int i = 0; i < 10; ++i) { if (value < GBA_LUX_LEVELS[i]) { m_luxLevel = i; break; } } emit luminanceValueChanged(m_luxValue); } void InputController::setupCam() { #ifdef BUILD_QT_MULTIMEDIA if (!m_camera) { m_camera = std::make_unique(); connect(m_camera.get(), &QCamera::statusChanged, this, &InputController::prepareCamSettings, Qt::QueuedConnection); } m_camera->setCaptureMode(QCamera::CaptureVideo); m_camera->setViewfinder(&m_videoDumper); m_camera->load(); #endif } #ifdef BUILD_QT_MULTIMEDIA void InputController::prepareCamSettings(QCamera::Status status) { if (status != QCamera::LoadedStatus || m_camera->state() == QCamera::ActiveState) { return; } #if (QT_VERSION >= QT_VERSION_CHECK(5, 5, 0)) QVideoFrame::PixelFormat format(QVideoFrame::Format_RGB32); QCameraViewfinderSettings settings; QSize size(1280, 720); auto cameraRes = m_camera->supportedViewfinderResolutions(settings); for (auto& cameraSize : cameraRes) { if (cameraSize.width() < m_image.w || cameraSize.height() < m_image.h) { continue; } if (cameraSize.width() <= size.width() && cameraSize.height() <= size.height()) { size = cameraSize; } } settings.setResolution(size); auto cameraFormats = m_camera->supportedViewfinderPixelFormats(settings); auto goodFormats = m_videoDumper.supportedPixelFormats(); bool goodFormatFound = false; for (const auto& goodFormat : goodFormats) { if (cameraFormats.contains(goodFormat)) { settings.setPixelFormat(goodFormat); format = goodFormat; goodFormatFound = true; break; } } if (!goodFormatFound) { LOG(QT, WARN) << "Could not find a valid camera format!"; for (const auto& format : cameraFormats) { LOG(QT, WARN) << "Camera supported format: " << QString::number(format); } } m_camera->setViewfinderSettings(settings); #endif m_camera->start(); } #endif void InputController::teardownCam() { #ifdef BUILD_QT_MULTIMEDIA if (m_camera) { m_camera->stop(); } #endif } void InputController::setCamera(const QByteArray& name) { #ifdef BUILD_QT_MULTIMEDIA bool needsRestart = false; if (m_camera) { needsRestart = m_camera->state() == QCamera::ActiveState; } m_camera = std::make_unique(name); connect(m_camera.get(), &QCamera::statusChanged, this, &InputController::prepareCamSettings, Qt::QueuedConnection); if (needsRestart) { setupCam(); } #endif }