/**************************************************************************** * * (c) 2009-2016 QGROUNDCONTROL PROJECT * * QGroundControl is licensed according to the terms in the file * COPYING.md in the root of the source code directory. * ****************************************************************************/ #include "SurveyMissionItem.h" #include "JsonHelper.h" #include "MissionController.h" #include "QGCGeo.h" #include "QGroundControlQmlGlobal.h" #include "QGCQGeoCoordinate.h" #include QGC_LOGGING_CATEGORY(SurveyMissionItemLog, "SurveyMissionItemLog") const char* SurveyMissionItem::jsonComplexItemTypeValue = "survey"; const char* SurveyMissionItem::_jsonGridObjectKey = "grid"; const char* SurveyMissionItem::_jsonGridAltitudeKey = "altitude"; const char* SurveyMissionItem::_jsonGridAltitudeRelativeKey = "relativeAltitude"; const char* SurveyMissionItem::_jsonGridAngleKey = "angle"; const char* SurveyMissionItem::_jsonGridSpacingKey = "spacing"; const char* SurveyMissionItem::_jsonGridEntryLocationKey = "entryLocation"; const char* SurveyMissionItem::_jsonTurnaroundDistKey = "turnAroundDistance"; const char* SurveyMissionItem::_jsonCameraTriggerDistanceKey = "cameraTriggerDistance"; const char* SurveyMissionItem::_jsonCameraTriggerInTurnaroundKey = "cameraTriggerInTurnaround"; const char* SurveyMissionItem::_jsonHoverAndCaptureKey = "hoverAndCapture"; const char* SurveyMissionItem::_jsonGroundResolutionKey = "groundResolution"; const char* SurveyMissionItem::_jsonFrontalOverlapKey = "imageFrontalOverlap"; const char* SurveyMissionItem::_jsonSideOverlapKey = "imageSideOverlap"; const char* SurveyMissionItem::_jsonCameraSensorWidthKey = "sensorWidth"; const char* SurveyMissionItem::_jsonCameraSensorHeightKey = "sensorHeight"; const char* SurveyMissionItem::_jsonCameraResolutionWidthKey = "resolutionWidth"; const char* SurveyMissionItem::_jsonCameraResolutionHeightKey = "resolutionHeight"; const char* SurveyMissionItem::_jsonCameraFocalLengthKey = "focalLength"; const char* SurveyMissionItem::_jsonCameraObjectKey = "camera"; const char* SurveyMissionItem::_jsonCameraNameKey = "name"; const char* SurveyMissionItem::_jsonManualGridKey = "manualGrid"; const char* SurveyMissionItem::_jsonCameraOrientationLandscapeKey = "orientationLandscape"; const char* SurveyMissionItem::_jsonFixedValueIsAltitudeKey = "fixedValueIsAltitude"; const char* SurveyMissionItem::_jsonRefly90DegreesKey = "refly90Degrees"; const char* SurveyMissionItem::settingsGroup = "Survey"; const char* SurveyMissionItem::manualGridName = "ManualGrid"; const char* SurveyMissionItem::gridAltitudeName = "GridAltitude"; const char* SurveyMissionItem::gridAltitudeRelativeName = "GridAltitudeRelative"; const char* SurveyMissionItem::gridAngleName = "GridAngle"; const char* SurveyMissionItem::gridSpacingName = "GridSpacing"; const char* SurveyMissionItem::gridEntryLocationName = "GridEntryLocation"; const char* SurveyMissionItem::turnaroundDistName = "TurnaroundDist"; const char* SurveyMissionItem::cameraTriggerDistanceName = "CameraTriggerDistance"; const char* SurveyMissionItem::cameraTriggerInTurnaroundName = "CameraTriggerInTurnaround"; const char* SurveyMissionItem::hoverAndCaptureName = "HoverAndCapture"; const char* SurveyMissionItem::groundResolutionName = "GroundResolution"; const char* SurveyMissionItem::frontalOverlapName = "FrontalOverlap"; const char* SurveyMissionItem::sideOverlapName = "SideOverlap"; const char* SurveyMissionItem::cameraSensorWidthName = "CameraSensorWidth"; const char* SurveyMissionItem::cameraSensorHeightName = "CameraSensorHeight"; const char* SurveyMissionItem::cameraResolutionWidthName = "CameraResolutionWidth"; const char* SurveyMissionItem::cameraResolutionHeightName = "CameraResolutionHeight"; const char* SurveyMissionItem::cameraFocalLengthName = "CameraFocalLength"; const char* SurveyMissionItem::cameraTriggerName = "CameraTrigger"; const char* SurveyMissionItem::cameraOrientationLandscapeName = "CameraOrientationLandscape"; const char* SurveyMissionItem::fixedValueIsAltitudeName = "FixedValueIsAltitude"; const char* SurveyMissionItem::cameraName = "Camera"; SurveyMissionItem::SurveyMissionItem(Vehicle* vehicle, QObject* parent) : ComplexMissionItem(vehicle, parent) , _sequenceNumber(0) , _dirty(false) , _mapPolygon(this) , _cameraOrientationFixed(false) , _missionCommandCount(0) , _refly90Degrees(false) , _ignoreRecalc(false) , _surveyDistance(0.0) , _cameraShots(0) , _coveredArea(0.0) , _timeBetweenShots(0.0) , _metaDataMap(FactMetaData::createMapFromJsonFile(QStringLiteral(":/json/Survey.SettingsGroup.json"), this)) , _manualGridFact (settingsGroup, _metaDataMap[manualGridName]) , _gridAltitudeFact (settingsGroup, _metaDataMap[gridAltitudeName]) , _gridAltitudeRelativeFact (settingsGroup, _metaDataMap[gridAltitudeRelativeName]) , _gridAngleFact (settingsGroup, _metaDataMap[gridAngleName]) , _gridSpacingFact (settingsGroup, _metaDataMap[gridSpacingName]) , _gridEntryLocationFact (settingsGroup, _metaDataMap[gridEntryLocationName]) , _turnaroundDistFact (settingsGroup, _metaDataMap[turnaroundDistName]) , _cameraTriggerDistanceFact (settingsGroup, _metaDataMap[cameraTriggerDistanceName]) , _cameraTriggerInTurnaroundFact (settingsGroup, _metaDataMap[cameraTriggerInTurnaroundName]) , _hoverAndCaptureFact (settingsGroup, _metaDataMap[hoverAndCaptureName]) , _groundResolutionFact (settingsGroup, _metaDataMap[groundResolutionName]) , _frontalOverlapFact (settingsGroup, _metaDataMap[frontalOverlapName]) , _sideOverlapFact (settingsGroup, _metaDataMap[sideOverlapName]) , _cameraSensorWidthFact (settingsGroup, _metaDataMap[cameraSensorWidthName]) , _cameraSensorHeightFact (settingsGroup, _metaDataMap[cameraSensorHeightName]) , _cameraResolutionWidthFact (settingsGroup, _metaDataMap[cameraResolutionWidthName]) , _cameraResolutionHeightFact (settingsGroup, _metaDataMap[cameraResolutionHeightName]) , _cameraFocalLengthFact (settingsGroup, _metaDataMap[cameraFocalLengthName]) , _cameraOrientationLandscapeFact (settingsGroup, _metaDataMap[cameraOrientationLandscapeName]) , _fixedValueIsAltitudeFact (settingsGroup, _metaDataMap[fixedValueIsAltitudeName]) , _cameraFact (settingsGroup, _metaDataMap[cameraName]) { _editorQml = "qrc:/qml/SurveyItemEditor.qml"; // NULL check since object creation during unit testing passes NULL for vehicle if (_vehicle && _vehicle->multiRotor()) { _turnaroundDistFact.setRawValue(0); } connect(&_gridSpacingFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(&_gridAngleFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(&_gridEntryLocationFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(&_turnaroundDistFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(&_cameraTriggerDistanceFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(&_cameraTriggerInTurnaroundFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(&_hoverAndCaptureFact, &Fact::valueChanged, this, &SurveyMissionItem::_generateGrid); connect(this, &SurveyMissionItem::refly90DegreesChanged, this, &SurveyMissionItem::_generateGrid); connect(&_gridAltitudeFact, &Fact::valueChanged, this, &SurveyMissionItem::_updateCoordinateAltitude); connect(&_gridAltitudeRelativeFact, &Fact::valueChanged, this, &SurveyMissionItem::_setDirty); // Signal to Qml when camera value changes so it can recalc connect(&_groundResolutionFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_frontalOverlapFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_sideOverlapFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraSensorWidthFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraSensorHeightFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraResolutionWidthFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraResolutionHeightFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraFocalLengthFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraOrientationLandscapeFact, &Fact::valueChanged, this, &SurveyMissionItem::_cameraValueChanged); connect(&_cameraTriggerDistanceFact, &Fact::valueChanged, this, &SurveyMissionItem::timeBetweenShotsChanged); connect(&_mapPolygon, &QGCMapPolygon::dirtyChanged, this, &SurveyMissionItem::_polygonDirtyChanged); connect(&_mapPolygon, &QGCMapPolygon::pathChanged, this, &SurveyMissionItem::_generateGrid); } void SurveyMissionItem::_setSurveyDistance(double surveyDistance) { if (!qFuzzyCompare(_surveyDistance, surveyDistance)) { _surveyDistance = surveyDistance; emit complexDistanceChanged(_surveyDistance); } } void SurveyMissionItem::_setCameraShots(int cameraShots) { if (_cameraShots != cameraShots) { _cameraShots = cameraShots; emit cameraShotsChanged(this->cameraShots()); } } void SurveyMissionItem::_setCoveredArea(double coveredArea) { if (!qFuzzyCompare(_coveredArea, coveredArea)) { _coveredArea = coveredArea; emit coveredAreaChanged(_coveredArea); } } void SurveyMissionItem::_clearInternal(void) { // Bug workaround while (_simpleGridPoints.count() > 1) { _simpleGridPoints.takeLast(); } emit gridPointsChanged(); _simpleGridPoints.clear(); _transectSegments.clear(); _missionCommandCount = 0; setDirty(true); emit specifiesCoordinateChanged(); emit lastSequenceNumberChanged(lastSequenceNumber()); } int SurveyMissionItem::lastSequenceNumber(void) const { return _sequenceNumber + _missionCommandCount; } void SurveyMissionItem::setCoordinate(const QGeoCoordinate& coordinate) { if (_coordinate != coordinate) { _coordinate = coordinate; emit coordinateChanged(_coordinate); } } void SurveyMissionItem::setDirty(bool dirty) { if (_dirty != dirty) { _dirty = dirty; emit dirtyChanged(_dirty); } } void SurveyMissionItem::save(QJsonArray& missionItems) { QJsonObject saveObject; saveObject[JsonHelper::jsonVersionKey] = 3; saveObject[VisualMissionItem::jsonTypeKey] = VisualMissionItem::jsonTypeComplexItemValue; saveObject[ComplexMissionItem::jsonComplexItemTypeKey] = jsonComplexItemTypeValue; saveObject[_jsonManualGridKey] = _manualGridFact.rawValue().toBool(); saveObject[_jsonFixedValueIsAltitudeKey] = _fixedValueIsAltitudeFact.rawValue().toBool(); saveObject[_jsonHoverAndCaptureKey] = _hoverAndCaptureFact.rawValue().toBool(); saveObject[_jsonRefly90DegreesKey] = _refly90Degrees; saveObject[_jsonCameraTriggerDistanceKey] = _cameraTriggerDistanceFact.rawValue().toDouble(); QJsonObject gridObject; gridObject[_jsonGridAltitudeKey] = _gridAltitudeFact.rawValue().toDouble(); gridObject[_jsonGridAltitudeRelativeKey] = _gridAltitudeRelativeFact.rawValue().toBool(); gridObject[_jsonGridAngleKey] = _gridAngleFact.rawValue().toDouble(); gridObject[_jsonGridSpacingKey] = _gridSpacingFact.rawValue().toDouble(); gridObject[_jsonGridEntryLocationKey] = _gridEntryLocationFact.rawValue().toDouble(); gridObject[_jsonTurnaroundDistKey] = _turnaroundDistFact.rawValue().toDouble(); saveObject[_jsonGridObjectKey] = gridObject; if (!_manualGridFact.rawValue().toBool()) { QJsonObject cameraObject; cameraObject[_jsonCameraNameKey] = _cameraFact.rawValue().toString(); cameraObject[_jsonCameraOrientationLandscapeKey] = _cameraOrientationLandscapeFact.rawValue().toBool(); cameraObject[_jsonCameraSensorWidthKey] = _cameraSensorWidthFact.rawValue().toDouble(); cameraObject[_jsonCameraSensorHeightKey] = _cameraSensorHeightFact.rawValue().toDouble(); cameraObject[_jsonCameraResolutionWidthKey] = _cameraResolutionWidthFact.rawValue().toDouble(); cameraObject[_jsonCameraResolutionHeightKey] = _cameraResolutionHeightFact.rawValue().toDouble(); cameraObject[_jsonCameraFocalLengthKey] = _cameraFocalLengthFact.rawValue().toDouble(); cameraObject[_jsonGroundResolutionKey] = _groundResolutionFact.rawValue().toDouble(); cameraObject[_jsonFrontalOverlapKey] = _frontalOverlapFact.rawValue().toInt(); cameraObject[_jsonSideOverlapKey] = _sideOverlapFact.rawValue().toInt(); saveObject[_jsonCameraObjectKey] = cameraObject; } // Polygon shape _mapPolygon.saveToJson(saveObject); missionItems.append(saveObject); } void SurveyMissionItem::setSequenceNumber(int sequenceNumber) { if (_sequenceNumber != sequenceNumber) { _sequenceNumber = sequenceNumber; emit sequenceNumberChanged(sequenceNumber); emit lastSequenceNumberChanged(lastSequenceNumber()); } } bool SurveyMissionItem::load(const QJsonObject& complexObject, int sequenceNumber, QString& errorString) { QJsonObject v2Object = complexObject; // We need to pull version first to determine what validation/conversion needs to be performed. QList versionKeyInfoList = { { JsonHelper::jsonVersionKey, QJsonValue::Double, true }, }; if (!JsonHelper::validateKeys(v2Object, versionKeyInfoList, errorString)) { return false; } int version = v2Object[JsonHelper::jsonVersionKey].toInt(); if (version != 2 && version != 3) { errorString = tr("%1 does not support this version of survey items").arg(qgcApp()->applicationName()); return false; } if (version == 2) { // Convert to v3 if (v2Object.contains(VisualMissionItem::jsonTypeKey) && v2Object[VisualMissionItem::jsonTypeKey].toString() == QStringLiteral("survey")) { v2Object[VisualMissionItem::jsonTypeKey] = VisualMissionItem::jsonTypeComplexItemValue; v2Object[ComplexMissionItem::jsonComplexItemTypeKey] = jsonComplexItemTypeValue; } } QList mainKeyInfoList = { { JsonHelper::jsonVersionKey, QJsonValue::Double, true }, { VisualMissionItem::jsonTypeKey, QJsonValue::String, true }, { ComplexMissionItem::jsonComplexItemTypeKey, QJsonValue::String, true }, { QGCMapPolygon::jsonPolygonKey, QJsonValue::Array, true }, { _jsonGridObjectKey, QJsonValue::Object, true }, { _jsonCameraObjectKey, QJsonValue::Object, false }, { _jsonCameraTriggerDistanceKey, QJsonValue::Double, true }, { _jsonManualGridKey, QJsonValue::Bool, true }, { _jsonFixedValueIsAltitudeKey, QJsonValue::Bool, true }, { _jsonHoverAndCaptureKey, QJsonValue::Bool, false }, { _jsonRefly90DegreesKey, QJsonValue::Bool, false }, }; if (!JsonHelper::validateKeys(v2Object, mainKeyInfoList, errorString)) { return false; } QString itemType = v2Object[VisualMissionItem::jsonTypeKey].toString(); QString complexType = v2Object[ComplexMissionItem::jsonComplexItemTypeKey].toString(); if (itemType != VisualMissionItem::jsonTypeComplexItemValue || complexType != jsonComplexItemTypeValue) { errorString = tr("%1 does not support loading this complex mission item type: %2:%3").arg(qgcApp()->applicationName()).arg(itemType).arg(complexType); return false; } _ignoreRecalc = true; _mapPolygon.clear(); setSequenceNumber(sequenceNumber); _manualGridFact.setRawValue (v2Object[_jsonManualGridKey].toBool(true)); _fixedValueIsAltitudeFact.setRawValue (v2Object[_jsonFixedValueIsAltitudeKey].toBool(true)); _gridAltitudeRelativeFact.setRawValue (v2Object[_jsonGridAltitudeRelativeKey].toBool(true)); _hoverAndCaptureFact.setRawValue (v2Object[_jsonHoverAndCaptureKey].toBool(false)); _refly90Degrees = v2Object[_jsonRefly90DegreesKey].toBool(false); QList gridKeyInfoList = { { _jsonGridAltitudeKey, QJsonValue::Double, true }, { _jsonGridAltitudeRelativeKey, QJsonValue::Bool, true }, { _jsonGridAngleKey, QJsonValue::Double, true }, { _jsonGridSpacingKey, QJsonValue::Double, true }, { _jsonGridEntryLocationKey, QJsonValue::Double, false }, { _jsonTurnaroundDistKey, QJsonValue::Double, true }, }; QJsonObject gridObject = v2Object[_jsonGridObjectKey].toObject(); if (!JsonHelper::validateKeys(gridObject, gridKeyInfoList, errorString)) { return false; } _gridAltitudeFact.setRawValue (gridObject[_jsonGridAltitudeKey].toDouble()); _gridAngleFact.setRawValue (gridObject[_jsonGridAngleKey].toDouble()); _gridSpacingFact.setRawValue (gridObject[_jsonGridSpacingKey].toDouble()); _turnaroundDistFact.setRawValue (gridObject[_jsonTurnaroundDistKey].toDouble()); _cameraTriggerDistanceFact.setRawValue (v2Object[_jsonCameraTriggerDistanceKey].toDouble()); if (gridObject.contains(_jsonGridEntryLocationKey)) { _gridEntryLocationFact.setRawValue(gridObject[_jsonGridEntryLocationKey].toDouble()); } else { _gridEntryLocationFact.setRawValue(_gridEntryLocationFact.rawDefaultValue()); } if (!_manualGridFact.rawValue().toBool()) { if (!v2Object.contains(_jsonCameraObjectKey)) { errorString = tr("%1 but %2 object is missing").arg("manualGrid = false").arg("camera"); return false; } QJsonObject cameraObject = v2Object[_jsonCameraObjectKey].toObject(); // Older code had typo on "imageSideOverlap" incorrectly being "imageSizeOverlap" QString incorrectImageSideOverlap = "imageSizeOverlap"; if (cameraObject.contains(incorrectImageSideOverlap)) { cameraObject[_jsonSideOverlapKey] = cameraObject[incorrectImageSideOverlap]; cameraObject.remove(incorrectImageSideOverlap); } QList cameraKeyInfoList = { { _jsonGroundResolutionKey, QJsonValue::Double, true }, { _jsonFrontalOverlapKey, QJsonValue::Double, true }, { _jsonSideOverlapKey, QJsonValue::Double, true }, { _jsonCameraSensorWidthKey, QJsonValue::Double, true }, { _jsonCameraSensorHeightKey, QJsonValue::Double, true }, { _jsonCameraResolutionWidthKey, QJsonValue::Double, true }, { _jsonCameraResolutionHeightKey, QJsonValue::Double, true }, { _jsonCameraFocalLengthKey, QJsonValue::Double, true }, { _jsonCameraNameKey, QJsonValue::String, true }, { _jsonCameraOrientationLandscapeKey, QJsonValue::Bool, true }, }; if (!JsonHelper::validateKeys(cameraObject, cameraKeyInfoList, errorString)) { return false; } _cameraFact.setRawValue(cameraObject[_jsonCameraNameKey].toString()); _cameraOrientationLandscapeFact.setRawValue(cameraObject[_jsonCameraOrientationLandscapeKey].toBool(true)); _groundResolutionFact.setRawValue (cameraObject[_jsonGroundResolutionKey].toDouble()); _frontalOverlapFact.setRawValue (cameraObject[_jsonFrontalOverlapKey].toInt()); _sideOverlapFact.setRawValue (cameraObject[_jsonSideOverlapKey].toInt()); _cameraSensorWidthFact.setRawValue (cameraObject[_jsonCameraSensorWidthKey].toDouble()); _cameraSensorHeightFact.setRawValue (cameraObject[_jsonCameraSensorHeightKey].toDouble()); _cameraResolutionWidthFact.setRawValue (cameraObject[_jsonCameraResolutionWidthKey].toDouble()); _cameraResolutionHeightFact.setRawValue (cameraObject[_jsonCameraResolutionHeightKey].toDouble()); _cameraFocalLengthFact.setRawValue (cameraObject[_jsonCameraFocalLengthKey].toDouble()); } // Polygon shape /// Load a polygon from json /// @param json Json object to load from /// @param required true: no polygon in object will generate error /// @param errorString Error string if return is false /// @return true: success, false: failure (errorString set) if (!_mapPolygon.loadFromJson(v2Object, true /* required */, errorString)) { _mapPolygon.clear(); return false; } _ignoreRecalc = false; _generateGrid(); return true; } double SurveyMissionItem::greatestDistanceTo(const QGeoCoordinate &other) const { double greatestDistance = 0.0; for (int i=0; i<_simpleGridPoints.count(); i++) { QGeoCoordinate currentCoord = _simpleGridPoints[i].value(); double distance = currentCoord.distanceTo(other); if (distance > greatestDistance) { greatestDistance = distance; } } return greatestDistance; } void SurveyMissionItem::_setExitCoordinate(const QGeoCoordinate& coordinate) { if (_exitCoordinate != coordinate) { _exitCoordinate = coordinate; emit exitCoordinateChanged(coordinate); } } bool SurveyMissionItem::specifiesCoordinate(void) const { return _mapPolygon.count() > 2; } void _calcCameraShots() { } void SurveyMissionItem::_convertTransectToGeo(const QList>& transectSegmentsNED, const QGeoCoordinate& tangentOrigin, QList>& transectSegmentsGeo) { transectSegmentsGeo.clear(); for (int i=0; i transectCoords; const QList& transectPoints = transectSegmentsNED[i]; for (int j=0; j>& transects) { QList> rgReversedTransects; for (int i=transects.count() - 1; i>=0; i--) { rgReversedTransects.append(transects[i]); } transects = rgReversedTransects; } /// Reverse the order of all points withing each transect, First point becomes last and so forth. void SurveyMissionItem::_reverseInternalTransectPoints(QList>& transects) { for (int i=0; i rgReversedCoords; QList& rgOriginalCoords = transects[i]; for (int j=rgOriginalCoords.count()-1; j>=0; j--) { rgReversedCoords.append(rgOriginalCoords[j]); } transects[i] = rgReversedCoords; } } /// Reorders the transects such that the first transect is the shortest distance to the specified coordinate /// and the first point within that transect is the shortest distance to the specified coordinate. /// @param distanceCoord Coordinate to measure distance against /// @param transects Transects to test and reorder void SurveyMissionItem::_optimizeTransectsForShortestDistance(const QGeoCoordinate& distanceCoord, QList>& transects) { double rgTransectDistance[4]; rgTransectDistance[0] = transects.first().first().distanceTo(distanceCoord); rgTransectDistance[1] = transects.first().last().distanceTo(distanceCoord); rgTransectDistance[2] = transects.last().first().distanceTo(distanceCoord); rgTransectDistance[3] = transects.last().last().distanceTo(distanceCoord); int shortestIndex = 0; double shortestDistance = rgTransectDistance[0]; for (int i=1; i<3; i++) { if (rgTransectDistance[i] < shortestDistance) { shortestIndex = i; shortestDistance = rgTransectDistance[i]; } } if (shortestIndex > 1) { // We need to reverse the order of segments _reverseTransectOrder(transects); } if (shortestIndex & 1) { // We need to reverse the points within each segment _reverseInternalTransectPoints(transects); } } void SurveyMissionItem::_appendGridPointsFromTransects(QList>& rgTransectSegments) { qCDebug(SurveyMissionItemLog) << "Entry point _appendGridPointsFromTransects" << rgTransectSegments.first().first(); for (int i=0; i& points, int index1, int index2) { QPointF temp = points[index1]; points[index1] = points[index2]; points[index2] = temp; } QList SurveyMissionItem::_convexPolygon(const QList& polygon) { // We use the Graham scan algorithem to convert the possibly concave polygon to a convex polygon // https://en.wikipedia.org/wiki/Graham_scan QList workPolygon(polygon); // First point must be lowest y-coordinate point for (int i=1; i angle) { _swapPoints(workPolygon, i, j); angle = _dp(workPolygon[0], workPolygon[j]); } } } // Perform the the Graham scan workPolygon.insert(0, workPolygon.last()); // Sentinel for algo stop int convexCount = 1; // Number of points on the convex hull. for (int i=2; i<=polygon.count(); i++) { while (_ccw(workPolygon[convexCount-1], workPolygon[convexCount], workPolygon[i]) <= 0) { if (convexCount > 1) { convexCount -= 1; } else if (i == polygon.count()) { break; } else { i++; } } convexCount++; _swapPoints(workPolygon, convexCount, i); } return workPolygon.mid(1, convexCount); } /// Returns true if the current grid angle generates north/south oriented transects bool SurveyMissionItem::_gridAngleIsNorthSouthTransects() { // Grid angle ranges from -360<->360 double gridAngle = qAbs(_gridAngleFact.rawValue().toDouble()); return gridAngle < 45.0 || (gridAngle > 360.0 - 45.0) || (gridAngle > 90.0 + 45.0 && gridAngle < 270.0 - 45.0); } void SurveyMissionItem::_adjustTransectsToEntryPointLocation(QList>& transects) { if (transects.count() == 0) { return; } // First determine what location the current entry point is at QGeoCoordinate& firstTransectEntry = transects.first().first(); QGeoCoordinate& firstTransectExit = transects.first().last(); QGeoCoordinate& lastTransectExit = transects.last().last(); bool northSouthTransects = _gridAngleIsNorthSouthTransects(); bool entryPointBottom; bool entryPointLeft; qCDebug(SurveyMissionItemLog) << "Original entry point" << transects.first().first(); qCDebug(SurveyMissionItemLog) << "northSouthTransects" << northSouthTransects; if (northSouthTransects) { double firstTransectAzimuth = firstTransectEntry.azimuthTo(firstTransectExit); qCDebug(SurveyMissionItemLog) << "firstTransectAzimuth" << firstTransectAzimuth; entryPointBottom = (firstTransectAzimuth >= 0.0 && firstTransectAzimuth < 90.0) || (firstTransectAzimuth > 270.0 && firstTransectAzimuth <= 360.0); qCDebug(SurveyMissionItemLog) << (entryPointBottom ? "Entry point is at bottom" : "Entry point is at top"); double entryToExitAzimuth = firstTransectEntry.azimuthTo(lastTransectExit); qCDebug(SurveyMissionItemLog) << "entryToExitAzimuth" << entryToExitAzimuth; entryPointLeft = entryToExitAzimuth <= 180.0; qCDebug(SurveyMissionItemLog) << (entryPointLeft ? "Entry point is at left" : "Entry point is at right"); } else { double firstTransectAzimuth = firstTransectEntry.azimuthTo(firstTransectExit); qCDebug(SurveyMissionItemLog) << "firstTransectAzimuth" << firstTransectAzimuth; entryPointLeft = firstTransectAzimuth <= 180.0; qCDebug(SurveyMissionItemLog) << (entryPointLeft ? "Entry point is at left" : "Entry point is at right"); double entryToExitAzimuth = firstTransectEntry.azimuthTo(lastTransectExit); qCDebug(SurveyMissionItemLog) << "entryToExitAzimuth" << entryToExitAzimuth; entryPointBottom = (entryToExitAzimuth >= 0.0 && entryToExitAzimuth < 90.0) || (entryToExitAzimuth > 270.0 && entryToExitAzimuth <= 360.0); qCDebug(SurveyMissionItemLog) << (entryPointBottom ? "Entry point is at bottom" : "Entry point is at top"); } // Now adjust the transects such that the entry point matches the requested location int entryLocation = _gridEntryLocationFact.rawValue().toInt(); bool reverseTransects; bool reversePoints; if (northSouthTransects) { reversePoints = ((entryLocation == EntryLocationTopLeft || entryLocation == EntryLocationTopRight) && entryPointBottom) || ((entryLocation == EntryLocationBottomLeft || entryLocation == EntryLocationBottomRight) && !entryPointBottom); reverseTransects = ((entryLocation == EntryLocationTopRight || entryLocation == EntryLocationBottomRight) && entryPointLeft) || ((entryLocation == EntryLocationTopLeft || entryLocation == EntryLocationBottomLeft) && !entryPointLeft); } else { reverseTransects = ((entryLocation == EntryLocationTopLeft || entryLocation == EntryLocationTopRight) && entryPointBottom) || ((entryLocation == EntryLocationBottomLeft || entryLocation == EntryLocationBottomRight) && !entryPointBottom); reversePoints = ((entryLocation == EntryLocationTopRight || entryLocation == EntryLocationBottomRight) && entryPointLeft) || ((entryLocation == EntryLocationTopLeft || entryLocation == EntryLocationBottomLeft) && !entryPointLeft); } if (reversePoints) { qCDebug(SurveyMissionItemLog) << "Reverse Points"; _reverseInternalTransectPoints(transects); } if (reverseTransects) { // The only way we should end up here is if there is a bug in the original grid line generation qCDebug(SurveyMissionItemLog) << "Not Reverse Transects"; //_reverseTransectOrder(transects); } qCDebug(SurveyMissionItemLog) << "Modified entry point" << transects.first().first(); } void SurveyMissionItem::_generateGrid(void) { if (_ignoreRecalc) { return; } if (_mapPolygon.count() < 3 || _gridSpacingFact.rawValue().toDouble() <= 0) { _clearInternal(); return; } _simpleGridPoints.clear(); _transectSegments.clear(); _reflyTransectSegments.clear(); QList polygonPoints; QList> transectSegments; // Convert polygon to NED QGeoCoordinate tangentOrigin = _mapPolygon.pathModel().value(0)->coordinate(); qCDebug(SurveyMissionItemLog) << "Convert polygon to NED - tangentOrigin" << tangentOrigin; for (int i=0; i<_mapPolygon.count(); i++) { double y, x, down; QGeoCoordinate vertex = _mapPolygon.pathModel().value(i)->coordinate(); if (i == 0) { // This avoids a nan calculation that comes out of convertGeoToNed x = y = 0; } else { convertGeoToNed(vertex, tangentOrigin, &y, &x, &down); } polygonPoints += QPointF(x, y); qCDebug(SurveyMissionItemLog) << "vertex:x:y" << vertex << polygonPoints.last().x() << polygonPoints.last().y(); } polygonPoints = _convexPolygon(polygonPoints); double coveredArea = 0.0; for (int i=0; i(); QGeoCoordinate coord2 = _simpleGridPoints[i].value(); surveyDistance += coord1.distanceTo(coord2); } _setSurveyDistance(surveyDistance); if (cameraShots == 0 && _triggerCamera()) { cameraShots = (int)ceil(surveyDistance / _triggerDistance()); } _setCameraShots(cameraShots); emit gridPointsChanged(); // Determine command count for lastSequenceNumber _missionCommandCount= 0; for (int i=0; i<_transectSegments.count(); i++) { const QList& transectSegment = _transectSegments[i]; _missionCommandCount += transectSegment.count(); // This accounts for all waypoints if (_hoverAndCaptureEnabled()) { // Internal camera trigger points are entry point, plus all points before exit point _missionCommandCount += transectSegment.count() - (_hasTurnaround() ? 2 : 0) - 1; } else if (_triggerCamera()) { _missionCommandCount += 2; // Camera on/off at entry/exit } } emit lastSequenceNumberChanged(lastSequenceNumber()); // Set exit coordinate if (_simpleGridPoints.count()) { QGeoCoordinate coordinate = _simpleGridPoints.first().value(); coordinate.setAltitude(_gridAltitudeFact.rawValue().toDouble()); setCoordinate(coordinate); QGeoCoordinate exitCoordinate = _simpleGridPoints.last().value(); exitCoordinate.setAltitude(_gridAltitudeFact.rawValue().toDouble()); _setExitCoordinate(exitCoordinate); } setDirty(true); } void SurveyMissionItem::_updateCoordinateAltitude(void) { _coordinate.setAltitude(_gridAltitudeFact.rawValue().toDouble()); _exitCoordinate.setAltitude(_gridAltitudeFact.rawValue().toDouble()); emit coordinateChanged(_coordinate); emit exitCoordinateChanged(_exitCoordinate); setDirty(true); } QPointF SurveyMissionItem::_rotatePoint(const QPointF& point, const QPointF& origin, double angle) { QPointF rotated; double radians = (M_PI / 180.0) * -angle; rotated.setX(((point.x() - origin.x()) * cos(radians)) - ((point.y() - origin.y()) * sin(radians)) + origin.x()); rotated.setY(((point.x() - origin.x()) * sin(radians)) + ((point.y() - origin.y()) * cos(radians)) + origin.y()); return rotated; } void SurveyMissionItem::_intersectLinesWithRect(const QList& lineList, const QRectF& boundRect, QList& resultLines) { QLineF topLine (boundRect.topLeft(), boundRect.topRight()); QLineF bottomLine (boundRect.bottomLeft(), boundRect.bottomRight()); QLineF leftLine (boundRect.topLeft(), boundRect.bottomLeft()); QLineF rightLine (boundRect.topRight(), boundRect.bottomRight()); for (int i=0; i& lineList, const QPolygonF& polygon, QList& resultLines) { for (int i=0; iP2 void SurveyMissionItem::_adjustLineDirection(const QList& lineList, QList& resultLines) { qreal firstAngle = 0; for (int i=0; i 1.0) { adjustedLine.setP1(line.p2()); adjustedLine.setP2(line.p1()); } else { adjustedLine = line; } resultLines += adjustedLine; } } double SurveyMissionItem::_clampGridAngle90(double gridAngle) { // Clamp grid angle to -90<->90. This prevents transects from being rotated to a reversed order. if (gridAngle > 90.0) { gridAngle -= 180.0; } else if (gridAngle < -90.0) { gridAngle += 180; } return gridAngle; } int SurveyMissionItem::_gridGenerator(const QList& polygonPoints, QList>& transectSegments, bool refly) { int cameraShots = 0; double gridAngle = _gridAngleFact.rawValue().toDouble(); double gridSpacing = _gridSpacingFact.rawValue().toDouble(); gridAngle = _clampGridAngle90(gridAngle); gridAngle += refly ? 90 : 0; qCDebug(SurveyMissionItemLog) << "Clamped grid angle" << gridAngle; qCDebug(SurveyMissionItemLog) << "SurveyMissionItem::_gridGenerator gridSpacing:gridAngle:refly" << gridSpacing << gridAngle << refly; transectSegments.clear(); // Convert polygon to bounding rect qCDebug(SurveyMissionItemLog) << "Polygon"; QPolygonF polygon; for (int i=0; i lineList; bool northSouthTransects = _gridAngleIsNorthSouthTransects(); int entryLocation = _gridEntryLocationFact.rawValue().toInt(); if (northSouthTransects) { qCDebug(SurveyMissionItemLog) << "Clamped grid angle" << gridAngle; if (entryLocation == EntryLocationTopLeft || entryLocation == EntryLocationBottomLeft) { // Generate transects from left to right qCDebug(SurveyMissionItemLog) << "Generate left to right"; float x = largeBoundRect.topLeft().x() - (gridSpacing / 2); while (x < largeBoundRect.bottomRight().x()) { float yTop = largeBoundRect.topLeft().y() - 100.0; float yBottom = largeBoundRect.bottomRight().y() + 100.0; lineList += QLineF(_rotatePoint(QPointF(x, yTop), center, gridAngle), _rotatePoint(QPointF(x, yBottom), center, gridAngle)); qCDebug(SurveyMissionItemLog) << "line(" << lineList.last().x1() << ", " << lineList.last().y1() << ")-(" << lineList.last().x2() <<", " << lineList.last().y2() << ")"; x += gridSpacing; } } else { // Generate transects from right to left qCDebug(SurveyMissionItemLog) << "Generate right to left"; float x = largeBoundRect.topRight().x() + (gridSpacing / 2); while (x > largeBoundRect.bottomLeft().x()) { float yTop = largeBoundRect.topRight().y() - 100.0; float yBottom = largeBoundRect.bottomLeft().y() + 100.0; lineList += QLineF(_rotatePoint(QPointF(x, yTop), center, gridAngle), _rotatePoint(QPointF(x, yBottom), center, gridAngle)); qCDebug(SurveyMissionItemLog) << "line(" << lineList.last().x1() << ", " << lineList.last().y1() << ")-(" << lineList.last().x2() <<", " << lineList.last().y2() << ")"; x -= gridSpacing; } } } else { gridAngle = _clampGridAngle90(gridAngle - 90.0); qCDebug(SurveyMissionItemLog) << "Clamped grid angle" << gridAngle; if (entryLocation == EntryLocationTopLeft || entryLocation == EntryLocationTopRight) { // Generate transects from top to bottom qCDebug(SurveyMissionItemLog) << "Generate top to bottom"; float y = largeBoundRect.bottomLeft().y() + (gridSpacing / 2); while (y > largeBoundRect.topRight().y()) { float xLeft = largeBoundRect.bottomLeft().x() - 100.0; float xRight = largeBoundRect.topRight().x() + 100.0; lineList += QLineF(_rotatePoint(QPointF(xLeft, y), center, gridAngle), _rotatePoint(QPointF(xRight, y), center, gridAngle)); qCDebug(SurveyMissionItemLog) << "y:xLeft:xRight" << y << xLeft << xRight << "line(" << lineList.last().x1() << ", " << lineList.last().y1() << ")-(" << lineList.last().x2() <<", " << lineList.last().y2() << ")"; y -= gridSpacing; } } else { // Generate transects from bottom to top qCDebug(SurveyMissionItemLog) << "Generate bottom to top"; float y = largeBoundRect.topLeft().y() - (gridSpacing / 2); while (y < largeBoundRect.bottomRight().y()) { float xLeft = largeBoundRect.topLeft().x() - 100.0; float xRight = largeBoundRect.bottomRight().x() + 100.0; lineList += QLineF(_rotatePoint(QPointF(xLeft, y), center, gridAngle), _rotatePoint(QPointF(xRight, y), center, gridAngle)); qCDebug(SurveyMissionItemLog) << "y:xLeft:xRight" << y << xLeft << xRight << "line(" << lineList.last().x1() << ", " << lineList.last().y1() << ")-(" << lineList.last().x2() <<", " << lineList.last().y2() << ")"; y += gridSpacing; } } } // Now intersect the lines with the polygon QList intersectLines; #if 1 _intersectLinesWithPolygon(lineList, polygon, intersectLines); #else // This is handy for debugging grid problems, not for release intersectLines = lineList; #endif // Make sure all lines are going to same direction. Polygon intersection leads to line which // can be in varied directions depending on the order of the intesecting sides. QList resultLines; _adjustLineDirection(intersectLines, resultLines); // Calc camera shots here if there are no images in turnaround if (_triggerCamera() && !_imagesEverywhere()) { for (int i=0; i transectPoints; const QLineF& line = resultLines[i]; float turnaroundPosition = _turnaroundDistance() / line.length(); if (i & 1) { transectLine = QLineF(line.p2(), line.p1()); } else { transectLine = QLineF(line.p1(), line.p2()); } // Build the points along the transect if (_hasTurnaround()) { transectPoints.append(transectLine.pointAt(-turnaroundPosition)); } // Polygon entry point transectPoints.append(transectLine.p1()); // For hover and capture we need points for each camera location if (_triggerCamera() && _hoverAndCaptureEnabled()) { if (_triggerDistance() < transectLine.length()) { int innerPoints = floor(transectLine.length() / _triggerDistance()); qCDebug(SurveyMissionItemLog) << "innerPoints" << innerPoints; float transectPositionIncrement = _triggerDistance() / transectLine.length(); for (int i=0; i& items, int seqNum, QGeoCoordinate& coord, CameraTriggerCode cameraTrigger, QObject* missionItemParent) { double altitude = _gridAltitudeFact.rawValue().toDouble(); bool altitudeRelative = _gridAltitudeRelativeFact.rawValue().toBool(); qCDebug(SurveyMissionItemLog) << "_appendWaypointToMission seq:trigger" << seqNum << (cameraTrigger != CameraTriggerNone); MissionItem* item = new MissionItem(seqNum++, MAV_CMD_NAV_WAYPOINT, altitudeRelative ? MAV_FRAME_GLOBAL_RELATIVE_ALT : MAV_FRAME_GLOBAL, cameraTrigger == CameraTriggerHoverAndCapture ? 1 : 0, // Hold time (1 second for hover and capture to settle vehicle before image is taken) 0.0, 0.0, std::numeric_limits::quiet_NaN(), // Yaw unchanged coord.latitude(), coord.longitude(), altitude, true, // autoContinue false, // isCurrentItem missionItemParent); items.append(item); switch (cameraTrigger) { case CameraTriggerOff: case CameraTriggerOn: item = new MissionItem(seqNum++, MAV_CMD_DO_SET_CAM_TRIGG_DIST, MAV_FRAME_MISSION, cameraTrigger == CameraTriggerOn ? _triggerDistance() : 0, 0, 0, 0, 0, 0, 0, // param 2-7 unused true, // autoContinue false, // isCurrentItem missionItemParent); items.append(item); break; case CameraTriggerHoverAndCapture: item = new MissionItem(seqNum++, MAV_CMD_IMAGE_START_CAPTURE, MAV_FRAME_MISSION, 0, // Camera ID, all cameras 0, // Interval (none) 1, // Take 1 photo -1, // Max horizontal resolution -1, // Max vertical resolution 0, 0, // param 6-7 not used true, // autoContinue false, // isCurrentItem missionItemParent); items.append(item); #if 0 // This generates too many commands. Pulling out for now, to see if image quality is still high enough. item = new MissionItem(seqNum++, MAV_CMD_NAV_DELAY, MAV_FRAME_MISSION, 0.5, // Delay in seconds, give some time for image to be taken -1, -1, -1, // No time 0, 0, 0, // Param 5-7 unused true, // autoContinue false, // isCurrentItem missionItemParent); items.append(item); #endif default: break; } return seqNum; } bool SurveyMissionItem::_nextTransectCoord(const QList& transectPoints, int pointIndex, QGeoCoordinate& coord) { if (pointIndex > transectPoints.count()) { qWarning() << "Bad grid generation"; return false; } coord = transectPoints[pointIndex]; return true; } /// Appends the mission items for the survey /// @param items Mission items are appended to this list /// @param missionItemParent Parent object for newly created MissionItem objects /// @param seqNum[in,out] Sequence number to start from /// @param hasRefly true: misison has a refly section /// @param buildRefly: true: build the refly section, false: build the first section /// @return false: Generation failed bool SurveyMissionItem::_appendMissionItemsWorker(QList& items, QObject* missionItemParent, int& seqNum, bool hasRefly, bool buildRefly) { qCDebug(SurveyMissionItemLog) << "hasTurnaround:triggerCamera:hoverAndCapture:imagesEverywhere:hasRefly:buildRefly" << _hasTurnaround() << _triggerCamera() << _hoverAndCaptureEnabled() << _imagesEverywhere() << hasRefly << buildRefly; QList>& transectSegments = buildRefly ? _reflyTransectSegments : _transectSegments; if (!buildRefly && _imagesEverywhere()) { // We are taking images in turnaround, so we start command once at beginning MissionItem* item = new MissionItem(seqNum++, MAV_CMD_DO_SET_CAM_TRIGG_DIST, MAV_FRAME_MISSION, _triggerDistance(), 0, 0, 0, 0, 0, 0, // param 2-7 unused true, // autoContinue false, // isCurrentItem missionItemParent); items.append(item); } for (int segmentIndex=0; segmentIndex& segment = transectSegments[segmentIndex]; qCDebug(SurveyMissionItemLog) << "segment.count" << segment.count(); if (_hasTurnaround()) { // Add entry turnaround point if (!_nextTransectCoord(segment, pointIndex++, coord)) { return false; } seqNum = _appendWaypointToMission(items, seqNum, coord, CameraTriggerNone, missionItemParent); } // Add polygon entry point if (!_nextTransectCoord(segment, pointIndex++, coord)) { return false; } cameraTrigger = _imagesEverywhere() || !_triggerCamera() ? CameraTriggerNone : (_hoverAndCaptureEnabled() ? CameraTriggerHoverAndCapture : CameraTriggerOn); seqNum = _appendWaypointToMission(items, seqNum, coord, cameraTrigger, missionItemParent); // Add internal hover and capture points if (_hoverAndCaptureEnabled()) { int lastHoverAndCaptureIndex = segment.count() - 1 - (_hasTurnaround() ? 1 : 0); qCDebug(SurveyMissionItemLog) << "lastHoverAndCaptureIndex" << lastHoverAndCaptureIndex; for (; pointIndex < lastHoverAndCaptureIndex; pointIndex++) { if (!_nextTransectCoord(segment, pointIndex, coord)) { return false; } seqNum = _appendWaypointToMission(items, seqNum, coord, CameraTriggerHoverAndCapture, missionItemParent); } } // Add polygon exit point if (!_nextTransectCoord(segment, pointIndex++, coord)) { return false; } cameraTrigger = _imagesEverywhere() || !_triggerCamera() ? CameraTriggerNone : (_hoverAndCaptureEnabled() ? CameraTriggerNone : CameraTriggerOff); seqNum = _appendWaypointToMission(items, seqNum, coord, cameraTrigger, missionItemParent); if (_hasTurnaround()) { // Add exit turnaround point if (!_nextTransectCoord(segment, pointIndex++, coord)) { return false; } seqNum = _appendWaypointToMission(items, seqNum, coord, CameraTriggerNone, missionItemParent); } qCDebug(SurveyMissionItemLog) << "last PointIndex" << pointIndex; } if (((hasRefly && buildRefly) || !hasRefly) && _imagesEverywhere()) { // Turn off camera at end of survey MissionItem* item = new MissionItem(seqNum++, MAV_CMD_DO_SET_CAM_TRIGG_DIST, MAV_FRAME_MISSION, 0.0, // trigger distance (off) 0, 0, 0, 0, 0, 0, // param 2-7 unused true, // autoContinue false, // isCurrentItem missionItemParent); items.append(item); } return true; } void SurveyMissionItem::appendMissionItems(QList& items, QObject* missionItemParent) { int seqNum = _sequenceNumber; if (!_appendMissionItemsWorker(items, missionItemParent, seqNum, _refly90Degrees, false /* buildRefly */)) { return; } if (_refly90Degrees) { _appendMissionItemsWorker(items, missionItemParent, seqNum, _refly90Degrees, true /* buildRefly */); } } int SurveyMissionItem::cameraShots(void) const { return _triggerCamera() ? _cameraShots : 0; } void SurveyMissionItem::_cameraValueChanged(void) { emit cameraValueChanged(); } double SurveyMissionItem::timeBetweenShots(void) const { return _cruiseSpeed == 0 ? 0 : _triggerDistance() / _cruiseSpeed; } void SurveyMissionItem::setMissionFlightStatus (MissionController::MissionFlightStatus_t& missionFlightStatus) { ComplexMissionItem::setMissionFlightStatus(missionFlightStatus); if (!qFuzzyCompare(_cruiseSpeed, missionFlightStatus.vehicleSpeed)) { _cruiseSpeed = missionFlightStatus.vehicleSpeed; emit timeBetweenShotsChanged(); } } void SurveyMissionItem::_setDirty(void) { setDirty(true); } bool SurveyMissionItem::hoverAndCaptureAllowed(void) const { return _vehicle->multiRotor() || _vehicle->vtol(); } double SurveyMissionItem::_triggerDistance(void) const { return _cameraTriggerDistanceFact.rawValue().toDouble(); } bool SurveyMissionItem::_triggerCamera(void) const { return _triggerDistance() > 0; } bool SurveyMissionItem::_imagesEverywhere(void) const { return _triggerCamera() && _cameraTriggerInTurnaroundFact.rawValue().toBool(); } bool SurveyMissionItem::_hoverAndCaptureEnabled(void) const { return hoverAndCaptureAllowed() && !_imagesEverywhere() && _triggerCamera() && _hoverAndCaptureFact.rawValue().toBool(); } bool SurveyMissionItem::_hasTurnaround(void) const { return _turnaroundDistance() > 0; } double SurveyMissionItem::_turnaroundDistance(void) const { return _turnaroundDistFact.rawValue().toDouble(); } void SurveyMissionItem::applyNewAltitude(double newAltitude) { _gridAltitudeFact.setRawValue(newAltitude); } void SurveyMissionItem::setRefly90Degrees(bool refly90Degrees) { if (refly90Degrees != _refly90Degrees) { _refly90Degrees = refly90Degrees; emit refly90DegreesChanged(refly90Degrees); } } void SurveyMissionItem::_polygonDirtyChanged(bool dirty) { if (dirty) { setDirty(true); } }