diff --git a/qgroundcontrol.qrc b/qgroundcontrol.qrc index f261e37..7e54e56 100644 --- a/qgroundcontrol.qrc +++ b/qgroundcontrol.qrc @@ -348,8 +348,10 @@ src/Vehicle/HygrometerFact.json src/Vehicle/GeneratorFact.json src/Vehicle/EFIFact.json + src/Gimbal/GimbalFact.json src/Settings/Video.SettingsGroup.json src/MissionManager/VTOLLandingPattern.FactMetaData.json + src/Settings/GimbalController.SettingsGroup.json src/comm/APMArduSubMockLink.params diff --git a/src/Gimbal/GimbalController.cc b/src/Gimbal/GimbalController.cc new file mode 100644 index 0000000..954098a --- /dev/null +++ b/src/Gimbal/GimbalController.cc @@ -0,0 +1,604 @@ +#include "GimbalController.h" + +#include "MAVLinkProtocol.h" +#include "Vehicle.h" +#include "QGCApplication.h" +#include "SettingsManager.h" + +#include + +QGC_LOGGING_CATEGORY(GimbalLog, "GimbalLog") + +const char* GimbalController::_gimbalFactGroupNamePrefix = "gimbal"; +const char* Gimbal::_absoluteRollFactName = "gimbalRoll"; +const char* Gimbal::_absolutePitchFactName = "gimbalPitch"; +const char* Gimbal::_bodyYawFactName = "gimbalYaw"; +const char* Gimbal::_absoluteYawFactName = "gimbalAzimuth"; +const char* Gimbal::_deviceIdFactName = "deviceId"; + +Gimbal::Gimbal() + : FactGroup(100, ":/json/Vehicle/GimbalFact.json") // No need to set parent as this will be deleted by gimbalController destructor +{ + _initFacts(); +} + +Gimbal::Gimbal(const Gimbal& other) + : FactGroup(100, ":/json/Vehicle/GimbalFact.json") // No need to set parent as this will be deleted by gimbalController destructor +{ + _initFacts(); + *this = other; +} + +const Gimbal& Gimbal::operator=(const Gimbal& other) +{ + _requestInformationRetries = other._requestInformationRetries; + _requestStatusRetries = other._requestStatusRetries; + _requestAttitudeRetries = other._requestAttitudeRetries; + _receivedInformation = other._receivedInformation; + _receivedStatus = other._receivedStatus; + _receivedAttitude = other._receivedAttitude; + _isComplete = other._isComplete; + _retracted = other._retracted; + _neutral = other._neutral; + _haveControl = other._haveControl; + _othersHaveControl = other._othersHaveControl; + _absoluteRollFact = other._absoluteRollFact; + _absolutePitchFact = other._absolutePitchFact; + _bodyYawFact = other._bodyYawFact; + _absoluteYawFact = other._absoluteYawFact; + _deviceIdFact = other._deviceIdFact; + _yawLock = other._yawLock; + _haveControl = other._haveControl; + _othersHaveControl = other._othersHaveControl; + + return *this; +} + +// To be called EXCLUSIVELY in Gimbal constructors +void Gimbal::_initFacts() +{ + _absoluteRollFact = Fact(0, _absoluteRollFactName, FactMetaData::valueTypeFloat); + _absolutePitchFact = Fact(0, _absolutePitchFactName, FactMetaData::valueTypeFloat); + _bodyYawFact = Fact(0, _bodyYawFactName, FactMetaData::valueTypeFloat); + _absoluteYawFact = Fact(0, _absoluteYawFactName, FactMetaData::valueTypeFloat); + _deviceIdFact = Fact(0, _deviceIdFactName, FactMetaData::valueTypeUint8); + + _addFact(&_absoluteRollFact, _absoluteRollFactName); + _addFact(&_absolutePitchFact, _absolutePitchFactName); + _addFact(&_bodyYawFact, _bodyYawFactName); + _addFact(&_absoluteYawFact, _absoluteYawFactName); + _addFact(&_deviceIdFact, _deviceIdFactName); + + _absoluteRollFact.setRawValue (0.0f); + _absolutePitchFact.setRawValue (0.0f); + _bodyYawFact.setRawValue (0.0f); + _absoluteYawFact.setRawValue (0.0f); + _deviceIdFact.setRawValue (0); +} + +GimbalController::GimbalController(MAVLinkProtocol* mavlink, Vehicle* vehicle) + : _mavlink(mavlink) + , _vehicle(vehicle) + , _activeGimbal(nullptr) +{ + QQmlEngine::setObjectOwnership(this, QQmlEngine::CppOwnership); + connect(_vehicle, &Vehicle::mavlinkMessageReceived, this, &GimbalController::_mavlinkMessageReceived); +} + +GimbalController::~GimbalController() +{ + _gimbals.clearAndDeleteContents(); +} + +void +GimbalController::setActiveGimbal(Gimbal* gimbal) +{ + if (!gimbal) { + qCDebug(GimbalLog) << "Set active gimbal: attempted to set a nullptr, returning"; + return; + } + + if (gimbal != _activeGimbal) { + qCDebug(GimbalLog) << "Set active gimbal: " << gimbal; + _activeGimbal = gimbal; + emit activeGimbalChanged(); + } +} + +void +GimbalController::_mavlinkMessageReceived(const mavlink_message_t& message) +{ + switch (message.msgid) { + case MAVLINK_MSG_ID_HEARTBEAT: + _handleHeartbeat(message); + break; + case MAVLINK_MSG_ID_GIMBAL_MANAGER_INFORMATION: + _handleGimbalManagerInformation(message); + break; + case MAVLINK_MSG_ID_GIMBAL_MANAGER_STATUS: + _handleGimbalManagerStatus(message); + break; + case MAVLINK_MSG_ID_GIMBAL_DEVICE_ATTITUDE_STATUS: + _handleGimbalDeviceAttitudeStatus(message); + break; + } +} + +void +GimbalController::_handleHeartbeat(const mavlink_message_t& message) +{ + if (!_potentialGimbalManagers.contains(message.compid)) { + qCDebug(GimbalLog) << "new potential gimbal manager component: " << message.compid; + } + + auto& gimbalManager = _potentialGimbalManagers[message.compid]; + + // Note that we are working over potential gimbal managers here, instead of potential gimbals. + // This is because we address the gimbal manager by compid, but a gimbal device might have an + // id different than the message compid it comes from. For more information see https://mavlink.io/en/services/gimbal_v2.html + if (!gimbalManager.receivedInformation && gimbalManager.requestGimbalManagerInformationRetries > 0) { + _requestGimbalInformation(message.compid); + --gimbalManager.requestGimbalManagerInformationRetries; + } +} + +void +GimbalController::_handleGimbalManagerInformation(const mavlink_message_t& message) +{ + + mavlink_gimbal_manager_information_t information; + mavlink_msg_gimbal_manager_information_decode(&message, &information); + + qCDebug(GimbalLog) << "_handleGimbalManagerInformation for gimbal device: " << information.gimbal_device_id << ", component id: " << message.compid; + + auto& gimbal = _potentialGimbals[information.gimbal_device_id]; + + if (information.gimbal_device_id != 0) { + gimbal.setDeviceId(information.gimbal_device_id); + } + + if (!gimbal._receivedInformation) { + qCDebug(GimbalLog) << "gimbal manager with compId: " << message.compid + << " is responsible for gimbal device: " << information.gimbal_device_id; + } + + gimbal._receivedInformation = true; + // It is important to flag our potential gimbal manager as well, so we stop requesting gimbal_manger_information message + auto& gimbalManager = _potentialGimbalManagers[message.compid]; + gimbalManager.receivedInformation = true; + + _checkComplete(gimbal, message.compid); +} + +void +GimbalController::_handleGimbalManagerStatus(const mavlink_message_t& message) +{ + + mavlink_gimbal_manager_status_t status; + mavlink_msg_gimbal_manager_status_decode(&message, &status); + + // qCDebug(GimbalLog) << "_handleGimbalManagerStatus for gimbal device: " << status.gimbal_device_id << ", component id: " << message.compid; + + auto& gimbal = _potentialGimbals[status.gimbal_device_id]; + + if (!status.gimbal_device_id) { + qCDebug(GimbalLog) << "gimbal manager with compId: " << message.compid + << " reported status of gimbal device id: " << status.gimbal_device_id << " which is not a valid gimbal device id"; + return; + } + + gimbal.setDeviceId(status.gimbal_device_id); + + // Only log this message once + if (!gimbal._receivedStatus) { + qCDebug(GimbalLog) << "_handleGimbalManagerStatus: gimbal manager with compId " << message.compid + << " is responsible for gimbal device " << status.gimbal_device_id; + } + + gimbal._receivedStatus = true; + + const bool haveControl = + (status.primary_control_sysid == _mavlink->getSystemId()) && + (status.primary_control_compid == _mavlink->getComponentId()); + + const bool othersHaveControl = !haveControl && + (status.primary_control_sysid != 0 && status.primary_control_compid != 0); + + if (gimbal.gimbalHaveControl() != haveControl) { + gimbal.setGimbalHaveControl(haveControl); + } + + if (gimbal.gimbalOthersHaveControl() != othersHaveControl) { + gimbal.setGimbalOthersHaveControl(othersHaveControl); + } + + _checkComplete(gimbal, message.compid); +} + +void +GimbalController::_handleGimbalDeviceAttitudeStatus(const mavlink_message_t& message) +{ + mavlink_gimbal_device_attitude_status_t attitude_status; + mavlink_msg_gimbal_device_attitude_status_decode(&message, &attitude_status); + + uint8_t gimbal_device_id_or_compid; + + // If gimbal_device_id is 0, we must take the compid of the message + if (attitude_status.gimbal_device_id == 0) { + gimbal_device_id_or_compid = message.compid; + + // If the gimbal_device_id field is set to 1-6, we must use this device id instead + } else if (attitude_status.gimbal_device_id <= 6) { + gimbal_device_id_or_compid = attitude_status.gimbal_device_id; + } + + // We do a reverse lookup here + auto gimbal_it = std::find_if(_potentialGimbals.begin(), _potentialGimbals.end(), + [&](auto& gimbal) { return gimbal.deviceId()->rawValue().toUInt() == gimbal_device_id_or_compid; }); + + if (gimbal_it == _potentialGimbals.end()) { + qCDebug(GimbalLog) << "_handleGimbalDeviceAttitudeStatus for unknown device id: " << gimbal_device_id_or_compid << " from component id: " << message.compid; + return; + } + + const bool yaw_in_vehicle_frame = _yawInVehicleFrame(attitude_status.flags); + + gimbal_it->setRetracted((attitude_status.flags & GIMBAL_DEVICE_FLAGS_RETRACT) > 0); + gimbal_it->setYawLock((attitude_status.flags & GIMBAL_DEVICE_FLAGS_YAW_LOCK) > 0); + gimbal_it->_neutral = (attitude_status.flags & GIMBAL_DEVICE_FLAGS_NEUTRAL) > 0; + + float roll, pitch, yaw; + mavlink_quaternion_to_euler(attitude_status.q, &roll, &pitch, &yaw); + + gimbal_it->setAbsoluteRoll(qRadiansToDegrees(roll)); + gimbal_it->setAbsolutePitch(qRadiansToDegrees(pitch)); + + if (yaw_in_vehicle_frame) { + float bodyYaw = qRadiansToDegrees(yaw); + float absoluteYaw = gimbal_it->bodyYaw()->rawValue().toFloat() + _vehicle->heading()->rawValue().toFloat(); + if (absoluteYaw > 180.0f) { + absoluteYaw -= 360.0f; + } + + gimbal_it->setBodyYaw(bodyYaw); + gimbal_it->setAbsoluteYaw(absoluteYaw); + + } else { + float absoluteYaw = qRadiansToDegrees(yaw); + float bodyYaw = gimbal_it->bodyYaw()->rawValue().toFloat() - _vehicle->heading()->rawValue().toFloat(); + if (bodyYaw < 180.0f) { + bodyYaw += 360.0f; + } + + gimbal_it->setBodyYaw(bodyYaw); + gimbal_it->setAbsoluteYaw(absoluteYaw); + } + + gimbal_it->_receivedAttitude = true; + + _checkComplete(*gimbal_it, message.compid); +} + +void +GimbalController::_requestGimbalInformation(uint8_t compid) +{ + qCDebug(GimbalLog) << "_requestGimbalInformation(" << compid << ")"; + + if(_vehicle) { + _vehicle->sendMavCommand(compid, + MAV_CMD_REQUEST_MESSAGE, + false /* no error */, + MAVLINK_MSG_ID_GIMBAL_MANAGER_INFORMATION); + } +} + +void +GimbalController::_checkComplete(Gimbal& gimbal, uint8_t compid) +{ + if (gimbal._isComplete) { + // Already complete, nothing to do. + return; + } + + if (!gimbal._receivedInformation && gimbal._requestInformationRetries > 0) { + _requestGimbalInformation(compid); + --gimbal._requestInformationRetries; + } + + if (!gimbal._receivedStatus && gimbal._requestStatusRetries > 0) { + _vehicle->sendMavCommand(compid, + MAV_CMD_SET_MESSAGE_INTERVAL, + false /* no error */, + MAVLINK_MSG_ID_GIMBAL_MANAGER_STATUS, + (gimbal._requestStatusRetries > 2) ? 0 : 5000000); // request default rate, if we don't succeed, last attempt is fixed 0.2 Hz instead + --gimbal._requestStatusRetries; + qCDebug(GimbalLog) << "attempt to set GIMBAL_MANAGER_STATUS message at" << (gimbal._requestStatusRetries > 2 ? "default rate" : "0.2 Hz") << "interval for device: " + << gimbal.deviceId()->rawValue().toUInt() << "compID: " << compid << ", retries remaining: " << gimbal._requestStatusRetries; + } + + if (!gimbal._receivedAttitude && gimbal._requestAttitudeRetries > 0 && + gimbal._receivedInformation && gimbal.deviceId()->rawValue().toUInt() != 0) { + // We request the attitude directly from the gimbal device component. + // We can only do that once we have received the gimbal manager information + // telling us which gimbal device it is responsible for. + _vehicle->sendMavCommand(gimbal.deviceId()->rawValue().toUInt(), + MAV_CMD_SET_MESSAGE_INTERVAL, + false /* no error */, + MAVLINK_MSG_ID_GIMBAL_DEVICE_ATTITUDE_STATUS, + 0 /* request default rate */); + + --gimbal._requestAttitudeRetries; + } + + if (!gimbal._receivedInformation || !gimbal._receivedStatus || !gimbal._receivedAttitude) { + // Not complete yet. + return; + } + + gimbal._isComplete = true; + + // If there is no current active gimbal, set this one as active + if (!_activeGimbal) { + setActiveGimbal(&gimbal); + } + + _gimbals.append(&gimbal); + // This is needed for new Gimbals telemetry to be available for the user to show in flyview telemetry panel + _vehicle->_addFactGroup(&gimbal, QStringLiteral("%1%2").arg(_gimbalFactGroupNamePrefix).arg(gimbal.deviceId()->rawValue().toUInt())); +} + +bool GimbalController::_tryGetGimbalControl() +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "_tryGetGimbalControl: active gimbal is nullptr, returning"; + return false; + } + // This means other component is in control, show popup + if (_activeGimbal->gimbalOthersHaveControl()) { + qCDebug(GimbalLog) << "Others in control, showing popup for user to confirm control.."; + emit showAcquireGimbalControlPopup(); + return false; + // This means nobody is in control, so we can adquire directly and attempt to control + } else if (!_activeGimbal->gimbalHaveControl()) { + qCDebug(GimbalLog) << "Nobody in control, acquiring control ourselves.."; + acquireGimbalControl(); + } + return true; +} + +bool GimbalController::_yawInVehicleFrame(uint32_t flags) +{ + if ((flags & GIMBAL_DEVICE_FLAGS_YAW_IN_VEHICLE_FRAME) > 0) { + return true; + } else if ((flags & GIMBAL_DEVICE_FLAGS_YAW_IN_EARTH_FRAME) > 0) { + return false; + } else { + // For backwards compatibility: if both new flags are 0, yaw lock defines the frame. + return (flags & GIMBAL_DEVICE_FLAGS_YAW_LOCK) == 0; + } +} + +void GimbalController::gimbalPitchStep(int direction) +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "gimbalStepPitch: active gimbal is nullptr, returning"; + return; + } + + if (_activeGimbal->yawLock()) { + sendPitchAbsoluteYaw(_activeGimbal->absolutePitch()->rawValue().toFloat() + direction, _activeGimbal->absoluteYaw()->rawValue().toFloat(), false); + } else { + sendPitchBodyYaw(_activeGimbal->absolutePitch()->rawValue().toFloat() + direction, _activeGimbal->bodyYaw()->rawValue().toFloat(), false); + } +} + +void GimbalController::gimbalYawStep(int direction) +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "gimbalStepPitch: active gimbal is nullptr, returning"; + return; + } + + if (_activeGimbal->yawLock()) { + sendPitchAbsoluteYaw(_activeGimbal->absolutePitch()->rawValue().toFloat(), _activeGimbal->absoluteYaw()->rawValue().toFloat() + direction, false); + } else { + sendPitchBodyYaw(_activeGimbal->absolutePitch()->rawValue().toFloat(), _activeGimbal->bodyYaw()->rawValue().toFloat() + direction, false); + } +} + +void GimbalController::centerGimbal() +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "gimbalYawStep: active gimbal is nullptr, returning"; + return; + } + sendPitchBodyYaw(0.0, 0.0); +} + +// Pan and tilt comes as +-(0-1) +void GimbalController::gimbalOnScreenControl(float panPct, float tiltPct, bool clickAndPoint, bool clickAndDrag, bool rateControl, bool retract, bool neutral, bool yawlock) +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "gimbalOnScreenControl: active gimbal is nullptr, returning"; + return; + } + // click and point, based on FOV + if (clickAndPoint) { + float hFov = qgcApp()->toolbox()->settingsManager()->gimbalControllerSettings()->CameraHFov()->rawValue().toFloat(); + float vFov = qgcApp()->toolbox()->settingsManager()->gimbalControllerSettings()->CameraVFov()->rawValue().toFloat(); + + float panIncDesired = panPct * hFov * 0.5f; + float tiltIncDesired = tiltPct * vFov * 0.5f; + + float panDesired = panIncDesired + _activeGimbal->bodyYaw()->rawValue().toFloat(); + float tiltDesired = tiltIncDesired + _activeGimbal->absolutePitch()->rawValue().toFloat(); + + if (_activeGimbal->yawLock()) { + sendPitchAbsoluteYaw(tiltDesired, panDesired + _vehicle->heading()->rawValue().toFloat(), false); + } else { + sendPitchBodyYaw(tiltDesired, panDesired, false); + } + + // click and drag, based on maximum speed + } else if (clickAndDrag) { + // Should send rate commands, but it seems for some reason it is not working on AP side. + // Pitch works ok but yaw doesn't stop, it keeps like inertia, like if it was buffering the messages. + // So we do a workaround with angle targets + float maxSpeed = qgcApp()->toolbox()->settingsManager()->gimbalControllerSettings()->CameraSlideSpeed()->rawValue().toFloat(); + + float panIncDesired = panPct * maxSpeed * 0.1f; + float tiltIncDesired = tiltPct * maxSpeed * 0.1f; + + float panDesired = panIncDesired + _activeGimbal->bodyYaw()->rawValue().toFloat(); + float tiltDesired = tiltIncDesired + _activeGimbal->absolutePitch()->rawValue().toFloat(); + + if (_activeGimbal->yawLock()) { + sendPitchAbsoluteYaw(tiltDesired, panDesired + _vehicle->heading()->rawValue().toFloat(), false); + } else { + sendPitchBodyYaw(tiltDesired, panDesired, false); + } + } +} + +void GimbalController::sendPitchBodyYaw(float pitch, float yaw, bool showError) { + if (!_tryGetGimbalControl()) { + return; + } + + // qDebug() << "sendPitch: " << pitch << " BodyYaw: " << yaw; + + unsigned flags = GIMBAL_MANAGER_FLAGS_ROLL_LOCK + | GIMBAL_MANAGER_FLAGS_PITCH_LOCK + | GIMBAL_MANAGER_FLAGS_YAW_IN_VEHICLE_FRAME; + + _vehicle->sendMavCommand( + _vehicle->compId(), + MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW, + showError, + pitch, + yaw, + 0, + 0, + flags, + 0, + _activeGimbal->deviceId()->rawValue().toUInt()); +} + +void GimbalController::sendPitchAbsoluteYaw(float pitch, float yaw, bool showError) { + if (!_tryGetGimbalControl()) { + return; + } + + if (yaw > 180.0f) { + yaw -= 360.0f; + } + + if (yaw < -180.0f) { + yaw += 360.0f; + } + + // qDebug() << "sendPitch: " << pitch << " absoluteYaw: " << yaw; + + unsigned flags = GIMBAL_MANAGER_FLAGS_ROLL_LOCK + | GIMBAL_MANAGER_FLAGS_PITCH_LOCK + | GIMBAL_MANAGER_FLAGS_YAW_LOCK + | GIMBAL_MANAGER_FLAGS_YAW_IN_EARTH_FRAME; + + _vehicle->sendMavCommand( + _vehicle->compId(), + MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW, + showError, + pitch, + yaw, + 0, + 0, + flags, + 0, + _activeGimbal->deviceId()->rawValue().toUInt()); +} + +void GimbalController::toggleGimbalRetracted(bool set) +{ + if (!_tryGetGimbalControl()) { + return; + } + + uint32_t flags = 0; + if (set) { + flags |= GIMBAL_DEVICE_FLAGS_RETRACT; + } else { + flags &= ~GIMBAL_DEVICE_FLAGS_RETRACT; + } + + sendPitchYawFlags(flags); +} + +void GimbalController::toggleGimbalYawLock(bool set) +{ + if (!_tryGetGimbalControl()) { + return; + } + + // Roll and pitch are usually "locked", so with horizon and not with aircraft. + uint32_t flags = GIMBAL_DEVICE_FLAGS_ROLL_LOCK | GIMBAL_DEVICE_FLAGS_PITCH_LOCK; + if (set) { + flags |= GIMBAL_DEVICE_FLAGS_YAW_LOCK; + } + + sendPitchYawFlags(flags); +} + +void GimbalController::sendPitchYawFlags(uint32_t flags) +{ + const bool yaw_in_vehicle_frame = _yawInVehicleFrame(flags); + + _vehicle->sendMavCommand( + _vehicle->compId(), + MAV_CMD_DO_GIMBAL_MANAGER_PITCHYAW, + true, + _activeGimbal->absolutePitch()->rawValue().toFloat(), + yaw_in_vehicle_frame ? _activeGimbal->bodyYaw()->rawValue().toFloat() : _activeGimbal->absoluteYaw()->rawValue().toFloat(), + static_cast(qQNaN()), + static_cast(qQNaN()), + flags, + 0, + _activeGimbal->deviceId()->rawValue().toUInt()); +} + +void GimbalController::acquireGimbalControl() +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "acquireGimbalControl: active gimbal is nullptr, returning"; + return; + } + _vehicle->sendMavCommand( + _vehicle->compId(), + MAV_CMD_DO_GIMBAL_MANAGER_CONFIGURE, + true, + _mavlink->getSystemId(), // Set us in primary control. + _mavlink->getComponentId(), // Set us in primary control + -1.f, // Leave secondary unchanged + -1.f, // Leave secondary unchanged + NAN, // Reserved + NAN, // Reserved + _activeGimbal->deviceId()->rawValue().toUInt()); +} + +void GimbalController::releaseGimbalControl() +{ + if (!_activeGimbal) { + qCDebug(GimbalLog) << "releaseGimbalControl: active gimbal is nullptr, returning"; + return; + } + _vehicle->sendMavCommand( + _vehicle->compId(), + MAV_CMD_DO_GIMBAL_MANAGER_CONFIGURE, + true, + -3.f, // Release primary control if we have control + -3.f, // Release primary control if we have control + -1.f, // Leave secondary control unchanged + -1.f, // Leave secondary control unchanged + NAN, // Reserved + NAN, // Reserved + _activeGimbal->deviceId()->rawValue().toUInt()); +} diff --git a/src/Gimbal/GimbalController.h b/src/Gimbal/GimbalController.h new file mode 100644 index 0000000..33bf769 --- /dev/null +++ b/src/Gimbal/GimbalController.h @@ -0,0 +1,157 @@ +/// @file GimbalController.h +/// @brief Class talking to gimbal managers based on the MAVLink gimbal v2 protocol. + +#pragma once + +#include +#include "Vehicle.h" +#include "QmlObjectListModel.h" + +Q_DECLARE_LOGGING_CATEGORY(GimbalLog) + +class MavlinkProtocol; + +class Gimbal : public FactGroup +{ + Q_OBJECT + + friend class GimbalController; // so it can set private members of gimbal, it is the only class that will need to modify them + +public: + Gimbal(); + Gimbal(const Gimbal& other); + const Gimbal& operator=(const Gimbal& other); + + Q_PROPERTY(Fact* absoluteRoll READ absoluteRoll CONSTANT) + Q_PROPERTY(Fact* absolutePitch READ absolutePitch CONSTANT) + Q_PROPERTY(Fact* bodyYaw READ bodyYaw CONSTANT) + Q_PROPERTY(Fact* absoluteYaw READ absoluteYaw CONSTANT) + Q_PROPERTY(Fact* deviceId READ deviceId CONSTANT) + Q_PROPERTY(bool yawLock READ yawLock NOTIFY yawLockChanged) + Q_PROPERTY(bool retracted READ retracted NOTIFY retractedChanged) + Q_PROPERTY(bool gimbalHaveControl READ gimbalHaveControl NOTIFY gimbalHaveControlChanged) + Q_PROPERTY(bool gimbalOthersHaveControl READ gimbalOthersHaveControl NOTIFY gimbalOthersHaveControlChanged) + + Fact* absoluteRoll() { return &_absoluteRollFact; } + Fact* absolutePitch() { return &_absolutePitchFact; } + Fact* bodyYaw() { return &_bodyYawFact; } + Fact* absoluteYaw() { return &_absoluteYawFact; } + Fact* deviceId() { return &_deviceIdFact; } + bool yawLock() const { return _yawLock; } + bool retracted() const { return _retracted; } + bool gimbalHaveControl() const { return _haveControl; } + bool gimbalOthersHaveControl() const { return _othersHaveControl; } + + void setAbsoluteRoll(float absoluteRoll) { _absoluteRollFact.setRawValue(absoluteRoll); } + void setAbsolutePitch(float absolutePitch) { _absolutePitchFact.setRawValue(absolutePitch); } + void setBodyYaw(float bodyYaw) { _bodyYawFact.setRawValue(bodyYaw); } + void setAbsoluteYaw(float absoluteYaw) { _absoluteYawFact.setRawValue(absoluteYaw); } + void setDeviceId(uint id) { _deviceIdFact.setRawValue(id); } + void setYawLock(bool yawLock) { _yawLock = yawLock; emit yawLockChanged(); } + void setRetracted(bool retracted) { _retracted = retracted; emit retractedChanged(); } + void setGimbalHaveControl(bool set) { _haveControl = set; emit gimbalHaveControlChanged(); } + void setGimbalOthersHaveControl(bool set) { _othersHaveControl = set; emit gimbalOthersHaveControlChanged(); } + + +signals: + void yawLockChanged(); + void retractedChanged(); + void gimbalHaveControlChanged(); + void gimbalOthersHaveControlChanged(); + +private: + void _initFacts(); // To be called EXCLUSIVELY in Gimbal constructors + + // Private members only accesed by friend class GimbalController + unsigned _requestInformationRetries = 3; + unsigned _requestStatusRetries = 6; + unsigned _requestAttitudeRetries = 3; + bool _receivedInformation = false; + bool _receivedStatus = false; + bool _receivedAttitude = false; + bool _isComplete = false; + bool _neutral = false; + + // Q_PROPERTIES + Fact _absoluteRollFact; + Fact _absolutePitchFact; + Fact _bodyYawFact; + Fact _absoluteYawFact; + Fact _deviceIdFact; // Component ID of gimbal device (or 1-6 for non-MAVLink gimbal) + bool _yawLock = false; + bool _retracted = false; + bool _haveControl = false; + bool _othersHaveControl = false; + + // Fact names + static const char* _absoluteRollFactName; + static const char* _absolutePitchFactName; + static const char* _bodyYawFactName; + static const char* _absoluteYawFactName; + static const char* _deviceIdFactName; +}; + +class GimbalController : public QObject +{ + Q_OBJECT +public: + GimbalController(MAVLinkProtocol* mavlink, Vehicle* vehicle); + ~GimbalController(); + + class GimbalManager { + public: + unsigned requestGimbalManagerInformationRetries = 6; + bool receivedInformation = false; + }; + + Q_PROPERTY(Gimbal* activeGimbal READ activeGimbal WRITE setActiveGimbal NOTIFY activeGimbalChanged) + Q_PROPERTY(QmlObjectListModel* gimbals READ gimbals CONSTANT) + + Gimbal* activeGimbal() { return _activeGimbal; } + QmlObjectListModel* gimbals() { return &_gimbals; } + + void setActiveGimbal(Gimbal* gimbal); + + void sendPitchYawFlags (uint32_t flags); + Q_INVOKABLE void gimbalOnScreenControl (float panpct, float tiltpct, bool clickAndPoint, bool clickAndDrag, bool rateControl, bool retract = false, bool neutral = false, bool yawlock = false); + Q_INVOKABLE void sendPitchBodyYaw (float pitch, float yaw, bool showError = true); + Q_INVOKABLE void sendPitchAbsoluteYaw (float pitch, float yaw, bool showError = true); + Q_INVOKABLE void toggleGimbalRetracted (bool set = false); + Q_INVOKABLE void toggleGimbalYawLock (bool set = false); + Q_INVOKABLE void acquireGimbalControl (); + Q_INVOKABLE void releaseGimbalControl (); + +public slots: + // These slots are conected with joysticks for button control + void gimbalYawLock (bool yawLock) { toggleGimbalYawLock(yawLock); } + Q_INVOKABLE void centerGimbal (); // Also used by qml + void gimbalPitchStep (int direction); + void gimbalYawStep (int direction); + +signals: + void activeGimbalChanged (); + void showAcquireGimbalControlPopup (); // This triggers a popup in QML asking the user for aproval to take control + +private slots: + void _mavlinkMessageReceived(const mavlink_message_t& message); + +private: + void _requestGimbalInformation (uint8_t compid); + void _handleHeartbeat (const mavlink_message_t& message); + void _handleGimbalManagerInformation (const mavlink_message_t& message); + void _handleGimbalManagerStatus (const mavlink_message_t& message); + void _handleGimbalDeviceAttitudeStatus (const mavlink_message_t& message); + void _checkComplete (Gimbal& gimbal, uint8_t compid); + bool _tryGetGimbalControl (); + bool _yawInVehicleFrame (uint32_t flags); + + MAVLinkProtocol* _mavlink = nullptr; + Vehicle* _vehicle = nullptr; + Gimbal* _activeGimbal = nullptr; + + QMap _potentialGimbalManagers; + QMap _potentialGimbals; + QmlObjectListModel _gimbals; + + static const char* _gimbalFactGroupNamePrefix; +}; diff --git a/src/Gimbal/GimbalFact.json b/src/Gimbal/GimbalFact.json new file mode 100644 index 0000000..868b242 --- /dev/null +++ b/src/Gimbal/GimbalFact.json @@ -0,0 +1,40 @@ +{ + "version": 1, + "fileType": "FactMetaData", + "QGC.MetaData.Facts": +[ +{ + "name": "gimbalRoll", + "shortDesc": "Gimbal Roll", + "type": "float", + "decimalPlaces": 1, + "units": "deg" +}, +{ + "name": "gimbalPitch", + "shortDesc": "Gimbal Pitch", + "type": "float", + "decimalPlaces": 1, + "units": "deg" +}, +{ + "name": "gimbalYaw", + "shortDesc": "Gimbal Yaw", + "type": "float", + "decimalPlaces": 1, + "units": "deg" +}, +{ + "name": "gimbalAzimuth", + "shortDesc": "Azimuth", + "type": "float", + "decimalPlaces": 1, + "units": "deg" +}, +{ + "name": "deviceId", + "shortDesc": "gimbal device Id", + "type": "uint8" +} +] +} diff --git a/src/Settings/GimbalController.SettingsGroup.json b/src/Settings/GimbalController.SettingsGroup.json new file mode 100644 index 0000000..2a47122 --- /dev/null +++ b/src/Settings/GimbalController.SettingsGroup.json @@ -0,0 +1,54 @@ +{ + "version": 1, + "fileType": "FactMetaData", + "QGC.MetaData.Facts": +[ +{ + "name": "EnableOnScreenControl", + "shortDesc": "Enable on Screen Camera Control", + "type": "bool", + "default": false +}, +{ + "name": "ControlType", + "shortDesc": "Type of on-screen control", + "type": "uint32", + "enumStrings": "Click to point, click and drag", + "enumValues": "0, 1", + "default": 0 +}, +{ + "name": "CameraVFov", + "shortDesc": "Vertical camera field of view", + "type": "uint32", + "default": 70, + "units": "deg" +}, +{ + "name": "CameraHFov", + "shortDesc": "Horizontal camera field of view", + "type": "uint32", + "default": 70, + "units": "deg" +}, +{ + "name": "CameraSlideSpeed", + "shortDesc": "Maximum gimbal speed on click and drag (deg/sec)", + "type": "uint32", + "default": 30, + "units": "deg/s" +}, +{ + "name": "showAzimuthIndicatorOnMap", + "shortDesc": "Show gimbal Azimuth indicator over vehicle icon in map", + "type": "bool", + "default": true +}, +{ + "name": "toolbarIndicatorShowAzimuth", + "shortDesc": "Show Azimuth instead of local yaw on top toolbar gimbal indicator", + "type": "bool", + "default": true +} +] +} \ No newline at end of file diff --git a/src/Settings/GimbalControllerSettings.cc b/src/Settings/GimbalControllerSettings.cc new file mode 100644 index 0000000..5d0ad7b --- /dev/null +++ b/src/Settings/GimbalControllerSettings.cc @@ -0,0 +1,32 @@ +/**************************************************************************** + * + * (c) 2009-2020 QGROUNDCONTROL PROJECT + * + * QGroundControl is licensed according to the terms in the file + * COPYING.md in the root of the source code directory. + * + ****************************************************************************/ + +#include "GimbalControllerSettings.h" +#include "QGCApplication.h" + +#include +#include +#include + +#ifndef QGC_DISABLE_UVC +#include +#endif + +DECLARE_SETTINGGROUP(GimbalController, "GimbalController") +{ + qmlRegisterUncreatableType("QGroundControl.SettingsManager", 1, 0, "GimbalControllerSettings", "Reference only"); +} + +DECLARE_SETTINGSFACT(GimbalControllerSettings, EnableOnScreenControl) +DECLARE_SETTINGSFACT(GimbalControllerSettings, ControlType) +DECLARE_SETTINGSFACT(GimbalControllerSettings, CameraVFov) +DECLARE_SETTINGSFACT(GimbalControllerSettings, CameraHFov) +DECLARE_SETTINGSFACT(GimbalControllerSettings, CameraSlideSpeed) +DECLARE_SETTINGSFACT(GimbalControllerSettings, showAzimuthIndicatorOnMap) +DECLARE_SETTINGSFACT(GimbalControllerSettings, toolbarIndicatorShowAzimuth) \ No newline at end of file diff --git a/src/Settings/GimbalControllerSettings.h b/src/Settings/GimbalControllerSettings.h new file mode 100644 index 0000000..b871d93 --- /dev/null +++ b/src/Settings/GimbalControllerSettings.h @@ -0,0 +1,28 @@ +/**************************************************************************** + * + * (c) 2009-2020 QGROUNDCONTROL PROJECT + * + * QGroundControl is licensed according to the terms in the file + * COPYING.md in the root of the source code directory. + * + ****************************************************************************/ +#pragma once + +#include "SettingsGroup.h" + +class GimbalControllerSettings : public SettingsGroup +{ + Q_OBJECT + +public: + GimbalControllerSettings(QObject* parent = nullptr); + DEFINE_SETTING_NAME_GROUP() + + DEFINE_SETTINGFACT(EnableOnScreenControl) + DEFINE_SETTINGFACT(ControlType) + DEFINE_SETTINGFACT(CameraVFov) + DEFINE_SETTINGFACT(CameraHFov) + DEFINE_SETTINGFACT(CameraSlideSpeed) + DEFINE_SETTINGFACT(showAzimuthIndicatorOnMap) + DEFINE_SETTINGFACT(toolbarIndicatorShowAzimuth) +}; \ No newline at end of file diff --git a/src/Settings/SettingsManager.cc b/src/Settings/SettingsManager.cc index 41d8ce9..36648cd 100644 --- a/src/Settings/SettingsManager.cc +++ b/src/Settings/SettingsManager.cc @@ -26,6 +26,7 @@ SettingsManager::SettingsManager(QGCApplication* app, QGCToolbox* toolbox) , _offlineMapsSettings (nullptr) , _firmwareUpgradeSettings (nullptr) , _adsbVehicleManagerSettings (nullptr) + , _gimbalControllerSettings (nullptr) #if !defined(NO_ARDUPILOT_DIALECT) , _apmMavlinkStreamRateSettings (nullptr) #endif @@ -52,6 +53,7 @@ void SettingsManager::setToolbox(QGCToolbox *toolbox) _offlineMapsSettings = new OfflineMapsSettings (this); _firmwareUpgradeSettings = new FirmwareUpgradeSettings (this); _adsbVehicleManagerSettings = new ADSBVehicleManagerSettings (this); + _gimbalControllerSettings = new GimbalControllerSettings (this); #if !defined(NO_ARDUPILOT_DIALECT) _apmMavlinkStreamRateSettings = new APMMavlinkStreamRateSettings(this); #endif diff --git a/src/Settings/SettingsManager.h b/src/Settings/SettingsManager.h index 3e8044a..8950c80 100644 --- a/src/Settings/SettingsManager.h +++ b/src/Settings/SettingsManager.h @@ -27,6 +27,7 @@ #include "APMMavlinkStreamRateSettings.h" #include "FirmwareUpgradeSettings.h" #include "ADSBVehicleManagerSettings.h" +#include "GimbalControllerSettings.h" #include #include "RemoteIDSettings.h" @@ -50,6 +51,7 @@ public: Q_PROPERTY(QObject* offlineMapsSettings READ offlineMapsSettings CONSTANT) Q_PROPERTY(QObject* firmwareUpgradeSettings READ firmwareUpgradeSettings CONSTANT) Q_PROPERTY(QObject* adsbVehicleManagerSettings READ adsbVehicleManagerSettings CONSTANT) + Q_PROPERTY(QObject* gimbalControllerSettings READ gimbalControllerSettings CONSTANT) #if !defined(NO_ARDUPILOT_DIALECT) Q_PROPERTY(QObject* apmMavlinkStreamRateSettings READ apmMavlinkStreamRateSettings CONSTANT) #endif @@ -69,6 +71,7 @@ public: OfflineMapsSettings* offlineMapsSettings (void) { return _offlineMapsSettings; } FirmwareUpgradeSettings* firmwareUpgradeSettings (void) { return _firmwareUpgradeSettings; } ADSBVehicleManagerSettings* adsbVehicleManagerSettings (void) { return _adsbVehicleManagerSettings; } + GimbalControllerSettings* gimbalControllerSettings (void) { return _gimbalControllerSettings; } #if !defined(NO_ARDUPILOT_DIALECT) APMMavlinkStreamRateSettings* apmMavlinkStreamRateSettings(void) { return _apmMavlinkStreamRateSettings; } #endif @@ -86,6 +89,7 @@ private: OfflineMapsSettings* _offlineMapsSettings; FirmwareUpgradeSettings* _firmwareUpgradeSettings; ADSBVehicleManagerSettings* _adsbVehicleManagerSettings; + GimbalControllerSettings* _gimbalControllerSettings; #if !defined(NO_ARDUPILOT_DIALECT) APMMavlinkStreamRateSettings* _apmMavlinkStreamRateSettings; #endif