/*=================================================================== QGroundControl Open Source Ground Control Station (c) 2009, 2010 QGROUNDCONTROL PROJECT This file is part of the QGROUNDCONTROL project QGROUNDCONTROL is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. QGROUNDCONTROL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with QGROUNDCONTROL. If not, see . ======================================================================*/ /** * @file * @brief MissionItem class * * @author Benjamin Knecht * @author Petri Tanskanen * */ #include #include #include "MissionItem.h" const MissionItem::MavCmd2Name_t MissionItem::_rgMavCmd2Name[_cMavCmd2Name] = { { MAV_CMD_NAV_WAYPOINT, "Waypoint" }, { MAV_CMD_NAV_LOITER_UNLIM, "Loiter" }, { MAV_CMD_NAV_LOITER_TURNS, "Loiter (turns)" }, { MAV_CMD_NAV_LOITER_TIME, "Loiter (unlimited)" }, { MAV_CMD_NAV_RETURN_TO_LAUNCH, "Return Home" }, { MAV_CMD_NAV_LAND, "Land" }, { MAV_CMD_NAV_TAKEOFF, "Takeoff" }, { MAV_CMD_CONDITION_DELAY, "Delay" }, { MAV_CMD_DO_JUMP, "Jump To Command" }, }; MissionItem::MissionItem(QObject* parent, int sequenceNumber, QGeoCoordinate coordinate, double param1, double param2, double param3, double param4, bool autocontinue, bool isCurrentItem, int frame, int action) : QObject(parent) , _sequenceNumber(sequenceNumber) , _coordinate(coordinate) , _yaw(param4) , _frame(frame) , _action(action) , _autocontinue(autocontinue) , _isCurrentItem(isCurrentItem) , _orbit(param3) , _param1(param1) , _param2(param2) , _reachedTime(0) { } MissionItem::MissionItem(const MissionItem& other) : QObject(NULL) { *this = other; } MissionItem::~MissionItem() { } const MissionItem& MissionItem::operator=(const MissionItem& other) { _sequenceNumber = other._sequenceNumber; _isCurrentItem = other._isCurrentItem; _coordinate = other._coordinate; _yaw = other._yaw; _frame = other._frame; _action = other._action; _autocontinue = other._autocontinue; _orbit = other._orbit; _param1 = other._param1; _param2 = other._param2; _reachedTime = other._reachedTime; return *this; } bool MissionItem::isNavigationType() { return (_action < MAV_CMD_NAV_LAST); } void MissionItem::save(QTextStream &saveStream) { QString position("%1\t%2\t%3"); position = position.arg(x(), 0, 'g', 18); position = position.arg(y(), 0, 'g', 18); position = position.arg(z(), 0, 'g', 18); QString parameters("%1\t%2\t%3\t%4"); parameters = parameters.arg(_param1, 0, 'g', 18).arg(_param2, 0, 'g', 18).arg(_orbit, 0, 'g', 18).arg(_yaw, 0, 'g', 18); // FORMAT: // as documented here: http://qgroundcontrol.org/waypoint_protocol saveStream << this->sequenceNumber() << "\t" << this->isCurrentItem() << "\t" << this->getFrame() << "\t" << this->getAction() << "\t" << parameters << "\t" << position << "\t" << this->getAutoContinue() << "\r\n"; //"\t" << this->getDescription() << "\r\n"; } bool MissionItem::load(QTextStream &loadStream) { const QStringList &wpParams = loadStream.readLine().split("\t"); if (wpParams.size() == 12) { setSequenceNumber(wpParams[0].toInt()); setIsCurrentItem(wpParams[1].toInt() == 1 ? true : false); _frame = (MAV_FRAME) wpParams[2].toInt(); _action = (MAV_CMD) wpParams[3].toInt(); _param1 = wpParams[4].toDouble(); _param2 = wpParams[5].toDouble(); _orbit = wpParams[6].toDouble(); setYaw(wpParams[7].toDouble()); setLatitude(wpParams[8].toDouble()); setLongitude(wpParams[9].toDouble()); setAltitude(wpParams[10].toDouble()); _autocontinue = (wpParams[11].toInt() == 1 ? true : false); return true; } return false; } void MissionItem::setSequenceNumber(int sequenceNumber) { _sequenceNumber = sequenceNumber; emit sequenceNumberChanged(_sequenceNumber); emit changed(this); } void MissionItem::setX(double x) { if (!isinf(x) && !isnan(x) && ((_frame == MAV_FRAME_LOCAL_NED) || (_frame == MAV_FRAME_LOCAL_ENU))) { setLatitude(x); } } void MissionItem::setY(double y) { if (!isinf(y) && !isnan(y) && ((_frame == MAV_FRAME_LOCAL_NED) || (_frame == MAV_FRAME_LOCAL_ENU))) { setLongitude(y); } } void MissionItem::setZ(double z) { if (!isinf(z) && !isnan(z) && ((_frame == MAV_FRAME_LOCAL_NED) || (_frame == MAV_FRAME_LOCAL_ENU))) { setAltitude(z); } } void MissionItem::setLatitude(double lat) { if (_coordinate.latitude() != lat && ((_frame == MAV_FRAME_GLOBAL) || (_frame == MAV_FRAME_GLOBAL_RELATIVE_ALT))) { _coordinate.setLatitude(lat); emit changed(this); emit coordinateChanged(coordinate()); } } void MissionItem::setLongitude(double lon) { if (_coordinate.longitude() != lon && ((_frame == MAV_FRAME_GLOBAL) || (_frame == MAV_FRAME_GLOBAL_RELATIVE_ALT))) { _coordinate.setLongitude(lon); emit changed(this); emit coordinateChanged(coordinate()); } } void MissionItem::setAltitude(double altitude) { if (_coordinate.altitude() != altitude && ((_frame == MAV_FRAME_GLOBAL) || (_frame == MAV_FRAME_GLOBAL_RELATIVE_ALT))) { _coordinate.setAltitude(altitude); emit changed(this); emit valueStringsChanged(valueStrings()); emit coordinateChanged(coordinate()); } } void MissionItem::setYaw(double yaw) { if (_yaw != yaw) { _yaw = yaw; emit yawChanged(yaw); emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setAction(int /*MAV_CMD*/ action) { if (_action != action) { _action = action; // Flick defaults according to WP type if (_action == MAV_CMD_NAV_TAKEOFF) { // We default to 15 degrees minimum takeoff pitch _param1 = 15.0; } emit changed(this); emit commandNameChanged(commandName()); emit commandChanged((MavlinkQmlSingleton::Qml_MAV_CMD)_action); emit specifiesCoordinateChanged(specifiesCoordinate()); emit valueLabelsChanged(valueLabels()); emit valueStringsChanged(valueStrings()); } } void MissionItem::setFrame(int /*MAV_FRAME*/ frame) { if (_frame != frame) { _frame = frame; emit changed(this); } } void MissionItem::setAutocontinue(bool autoContinue) { if (_autocontinue != autoContinue) { _autocontinue = autoContinue; emit changed(this); } } void MissionItem::setIsCurrentItem(bool isCurrentItem) { if (_isCurrentItem != isCurrentItem) { _isCurrentItem = isCurrentItem; emit isCurrentItemChanged(isCurrentItem); } } void MissionItem::setAcceptanceRadius(double radius) { if (_param2 != radius) { _param2 = radius; emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setParam1(double param1) { //// // qDebug() << "SENDER:" << QObject::sender(); //// // qDebug() << "PARAM1 SET REQ:" << param1; if (_param1 != param1) { _param1 = param1; emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setParam2(double param2) { if (_param2 != param2) { _param2 = param2; emit valueStringsChanged(valueStrings()); emit changed(this); } } void MissionItem::setParam3(double param3) { if (_orbit != param3) { _orbit = param3; emit valueStringsChanged(valueStrings()); emit changed(this); } } void MissionItem::setParam4(double param4) { if (_yaw != param4) { _yaw = param4; emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setParam5(double param5) { if (_coordinate.latitude() != param5) { _coordinate.setLatitude(param5); emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setParam6(double param6) { if (_coordinate.longitude() != param6) { _coordinate.setLongitude(param6); emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setParam7(double param7) { if (_coordinate.altitude() != param7) { _coordinate.setAltitude(param7); emit valueStringsChanged(valueStrings()); emit changed(this); } } void MissionItem::setLoiterOrbit(double orbit) { if (_orbit != orbit) { _orbit = orbit; emit valueStringsChanged(valueStrings()); emit changed(this); } } void MissionItem::setHoldTime(int holdTime) { if (_param1 != holdTime) { _param1 = holdTime; emit valueStringsChanged(valueStrings()); emit changed(this); } } void MissionItem::setHoldTime(double holdTime) { if (_param1 != holdTime) { _param1 = holdTime; emit changed(this); emit valueStringsChanged(valueStrings()); } } void MissionItem::setTurns(int turns) { if (_param1 != turns) { _param1 = turns; emit changed(this); emit valueStringsChanged(valueStrings()); } } bool MissionItem::specifiesCoordinate(void) const { switch (_action) { case MAV_CMD_NAV_WAYPOINT: case MAV_CMD_NAV_LOITER_UNLIM: case MAV_CMD_NAV_LOITER_TURNS: case MAV_CMD_NAV_LOITER_TIME: case MAV_CMD_NAV_LAND: case MAV_CMD_NAV_TAKEOFF: return true; default: return false; } } QString MissionItem::commandName(void) { QString type; switch (_action) { case MAV_CMD_NAV_WAYPOINT: type = "Waypoint"; break; case MAV_CMD_NAV_LOITER_UNLIM: case MAV_CMD_NAV_LOITER_TURNS: case MAV_CMD_NAV_LOITER_TIME: type = "Loiter"; break; case MAV_CMD_NAV_RETURN_TO_LAUNCH: type = "Return Home"; break; case MAV_CMD_NAV_LAND: type = "Land"; break; case MAV_CMD_NAV_TAKEOFF: type = "Takeoff"; break; case MAV_CMD_CONDITION_DELAY: type = "Delay"; break; case MAV_CMD_DO_JUMP: type = "Jump To Command"; break; default: type = QString("Unknown (%1)").arg(_action); break; } return type; } QStringList MissionItem::valueLabels(void) { QStringList labels; switch (_action) { case MAV_CMD_NAV_WAYPOINT: if (getFrame() == MAV_FRAME_GLOBAL_RELATIVE_ALT) { labels << "Alt (rel):"; } else { labels << "Alt:"; } labels << "Heading:" << "Radius:" << "Hold:"; break; case MAV_CMD_NAV_LOITER_UNLIM: labels << "Heading:" << "Radius:"; break; case MAV_CMD_NAV_LOITER_TURNS: labels << "Heading:" << "Radius:"<< "Turns:"; break; case MAV_CMD_NAV_LOITER_TIME: labels << "Heading:" << "Radius:" << "Seconds:"; break; case MAV_CMD_NAV_RETURN_TO_LAUNCH: break; case MAV_CMD_NAV_LAND: if (getFrame() == MAV_FRAME_GLOBAL_RELATIVE_ALT) { labels << "Alt (rel):"; } else { labels << "Alt:"; } labels << "Heading:"; break; case MAV_CMD_NAV_TAKEOFF: if (getFrame() == MAV_FRAME_GLOBAL_RELATIVE_ALT) { labels << "Alt (rel):"; } else { labels << "Alt:"; } labels << "Heading:" << "Pitch:"; break; case MAV_CMD_CONDITION_DELAY: labels << "Seconds:"; break; case MAV_CMD_DO_JUMP: labels << "Jump to:" << "Repeat:"; break; default: break; } return labels; } QString MissionItem::_oneDecimalString(double value) { return QString("%1").arg(value, 0 /* min field width */, 'f' /* format */, 1 /* precision */); } QStringList MissionItem::valueStrings(void) { QStringList list; switch (_action) { case MAV_CMD_NAV_WAYPOINT: list << _oneDecimalString(_coordinate.altitude()) << _oneDecimalString(_yaw * (180.0 / M_PI)) << _oneDecimalString(_param2) << _oneDecimalString(_param1); break; case MAV_CMD_NAV_LOITER_UNLIM: list << _oneDecimalString(_yaw * (180.0 / M_PI)) << _oneDecimalString(_orbit); break; case MAV_CMD_NAV_LOITER_TURNS: list << _oneDecimalString(_yaw * (180.0 / M_PI)) << _oneDecimalString(_orbit) << _oneDecimalString(_turns); break; case MAV_CMD_NAV_LOITER_TIME: list << _oneDecimalString(_yaw * (180.0 / M_PI)) << _oneDecimalString(_orbit) << _oneDecimalString(_param1); break; case MAV_CMD_NAV_RETURN_TO_LAUNCH: break; case MAV_CMD_NAV_LAND: list << _oneDecimalString(_coordinate.altitude()) << _oneDecimalString(_yaw * (180.0 / M_PI)); break; case MAV_CMD_NAV_TAKEOFF: list << _oneDecimalString(_coordinate.altitude()) << _oneDecimalString(_yaw * (180.0 / M_PI)) << _oneDecimalString(_param1); break; case MAV_CMD_CONDITION_DELAY: list << _oneDecimalString(_param1); break; case MAV_CMD_DO_JUMP: list << _oneDecimalString(_param1) << _oneDecimalString(_param2); break; default: break; } return list; } void MissionItem::setCoordinate(const QGeoCoordinate& coordinate) { _coordinate = coordinate; emit coordinateChanged(coordinate); emit changed(this); } QStringList MissionItem::commandNames(void) { QStringList list; for (int i=0; i<_cMavCmd2Name; i++) { list += _rgMavCmd2Name[i].name; } return list; } int MissionItem::commandByIndex(void) { for (int i=0; i<_cMavCmd2Name; i++) { if (_rgMavCmd2Name[i].command == _action) { return i; } } return -1; } void MissionItem::setCommandByIndex(int index) { if (index < 0 || index >= _cMavCmd2Name) { qWarning() << "Invalid index" << index; return; } setCommand((MavlinkQmlSingleton::Qml_MAV_CMD)_rgMavCmd2Name[index].command); }