MissionController.cc 6.44 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
/*=====================================================================

QGroundControl Open Source Ground Control Station

(c) 2009, 2015 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>

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 <http://www.gnu.org/licenses/>.

======================================================================*/

#include "MissionController.h"
#include "ScreenToolsController.h"
#include "MultiVehicleManager.h"
#include "MissionManager.h"
#include "CoordinateVector.h"

MissionController::MissionController(QObject *parent)
    : QObject(parent)
    , _missionItems(NULL)
    , _activeVehicle(NULL)
{
    MultiVehicleManager* multiVehicleMgr = MultiVehicleManager::instance();
    
    connect(multiVehicleMgr, &MultiVehicleManager::activeVehicleChanged, this, &MissionController::_activeVehicleChanged);
    
    Vehicle* activeVehicle = multiVehicleMgr->activeVehicle();
    if (activeVehicle) {
        _activeVehicleChanged(activeVehicle);
    } else {
        _newMissionItemsAvailable();
    }
}

MissionController::~MissionController()
{
}

void MissionController::_newMissionItemsAvailable(void)
{
    if (_missionItems) {
        _missionItems->deleteLater();
    }
    
    MissionManager* missionManager = NULL;
    Vehicle* activeVehicle = MultiVehicleManager::instance()->activeVehicle();
    if (activeVehicle) {
        missionManager = activeVehicle->missionManager();
    }

    if (!missionManager || missionManager->inProgress()) {
        _missionItems = new QmlObjectListModel(this);
    } else {
        _missionItems = missionManager->copyMissionItems();
    }

    _initAllMissionItems();
}

void MissionController::_recalcWaypointLines(void)
{
74 75
    int firstIndex = _homePositionValid  ? 0 : 1;

76
    _waypointLines.clear();
77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93

    if (firstIndex < _missionItems->count()) {
        bool firstCoordinateItem = true;
        MissionItem* lastCoordinateItem = qobject_cast<MissionItem*>(_missionItems->get(firstIndex));

        for (int i=firstIndex; i<_missionItems->count(); i++) {
            MissionItem* item = qobject_cast<MissionItem*>(_missionItems->get(i));

            if (item->specifiesCoordinate()) {
                if (firstCoordinateItem) {
                    if (item->command() == MavlinkQmlSingleton::MAV_CMD_NAV_TAKEOFF && _homePositionValid) {
                        // The first coordinate we hit is a takeoff command so link back to home position if we have one
                        _waypointLines.append(new CoordinateVector(qobject_cast<MissionItem*>(_missionItems->get(0))->coordinate(), item->coordinate()));
                    } else {
                        // First coordiante is not a takeoff command, it does not link backwards to anything
                    }
                    firstCoordinateItem = false;
94
                } else {
95 96
                    // Subsequent coordinate items link to last coordinate item
                    _waypointLines.append(new CoordinateVector(lastCoordinateItem->coordinate(), item->coordinate()));
97
                }
98
                lastCoordinateItem = item;
99 100 101 102 103 104 105 106 107 108
            }
        }
    }
    
    emit waypointLinesChanged();
}

// This will update the child item hierarchy
void MissionController::_recalcChildItems(void)
{
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
    int firstIndex = _homePositionValid  ? 0 : 1;

    if (_missionItems->count() > firstIndex) {
        MissionItem* currentParentItem = qobject_cast<MissionItem*>(_missionItems->get(firstIndex));

        currentParentItem->childItems()->clear();

        for (int i=firstIndex+1; i<_missionItems->count(); i++) {
            MissionItem* item = qobject_cast<MissionItem*>(_missionItems->get(i));

            // Set up non-coordinate item child hierarchy
            if (item->specifiesCoordinate()) {
                item->childItems()->clear();
                currentParentItem = item;
            } else {
                currentParentItem->childItems()->append(item);
            }
126 127 128 129
        }
    }
}

130 131 132 133 134 135 136 137 138 139 140
// This will update the sequence numbers to be sequential starting from 0
void MissionController::_recalcSequence(void)
{
    for (int i=0; i<_missionItems->count(); i++) {
        MissionItem* item = qobject_cast<MissionItem*>(_missionItems->get(i));

        // Setup ascending sequence numbers
        item->setSequenceNumber(i);
    }
}

141 142
void MissionController::_recalcAll(void)
{
143
    _recalcSequence();
144 145 146 147 148 149 150 151 152 153
    _recalcChildItems();
    _recalcWaypointLines();
}

/// Initializes a new set of mission items which may have come from the vehicle or have been loaded from a file
void MissionController::_initAllMissionItems(void)
{
    // Add the home position item to the front
    MissionItem* homeItem = new MissionItem(this);
    homeItem->setHomePositionSpecialCase(true);
154
    homeItem->setHomePositionValid(false);
155
    homeItem->setCommand(MavlinkQmlSingleton::MAV_CMD_NAV_WAYPOINT);
156 157
    homeItem->setLatitude(47.3769);
    homeItem->setLongitude(8.549444);
158 159
    _missionItems->insert(0, homeItem);
    
160
    _recalcAll();
161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
    
    emit missionItemsChanged();
}

void MissionController::_activeVehicleChanged(Vehicle* activeVehicle)
{
    if (_activeVehicle) {
        MissionManager* missionManager = _activeVehicle->missionManager();
        disconnect(missionManager, &MissionManager::newMissionItemsAvailable,   this, &MissionController::_newMissionItemsAvailable);
        _activeVehicle = NULL;
    }
    
    _activeVehicle = activeVehicle;
    
    if (_activeVehicle) {
        MissionManager* missionManager = activeVehicle->missionManager();
        connect(missionManager, &MissionManager::newMissionItemsAvailable,  this, &MissionController::_newMissionItemsAvailable);
        _newMissionItemsAvailable();
    }
}
181 182 183 184 185 186 187 188

void MissionController::setHomePositionValid(bool homePositionValid)
{
    _homePositionValid = homePositionValid;
    qobject_cast<MissionItem*>(_missionItems->get(0))->setHomePositionValid(homePositionValid);

    emit homePositionValidChanged(_homePositionValid);
}