WaypointList.cc 13.2 KB
Newer Older
pixhawk's avatar
pixhawk committed
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
/*=====================================================================

PIXHAWK Micro Air Vehicle Flying Robotics Toolkit

(c) 2009, 2010 PIXHAWK PROJECT  <http://pixhawk.ethz.ch>

This file is part of the PIXHAWK project

    PIXHAWK 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.

    PIXHAWK 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 PIXHAWK. If not, see <http://www.gnu.org/licenses/>.

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

/**
 * @file
 *   @brief Waypoint list widget
 *
 *   @author Lorenz Meier <mavteam@student.ethz.ch>
 *   @author Benjamin Knecht <mavteam@student.ethz.ch>
30
 *   @author Petri Tanskanen <mavteam@student.ethz.ch>
pixhawk's avatar
pixhawk committed
31 32 33 34 35 36 37 38 39 40 41 42 43
 *
 */

#include "WaypointList.h"
#include "ui_WaypointList.h"
#include <UASInterface.h>
#include <UASManager.h>
#include <QDebug>
#include <QFileDialog>

WaypointList::WaypointList(QWidget *parent, UASInterface* uas) :
        QWidget(parent),
        uas(NULL),
44 45 46 47
        mavX(0.0),
        mavY(0.0),
        mavZ(0.0),
        mavYaw(0.0),
pixhawk's avatar
pixhawk committed
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
        m_ui(new Ui::WaypointList)
{
    m_ui->setupUi(this);

    listLayout = new QVBoxLayout(m_ui->listWidget);
    listLayout->setSpacing(6);
    listLayout->setMargin(0);
    listLayout->setAlignment(Qt::AlignTop);
    m_ui->listWidget->setLayout(listLayout);

    wpViews = QMap<Waypoint*, WaypointView*>();

    this->uas = NULL;

    // ADD WAYPOINT
    // Connect add action, set right button icon and connect action to this class
    connect(m_ui->addButton, SIGNAL(clicked()), m_ui->actionAddWaypoint, SIGNAL(triggered()));
    connect(m_ui->actionAddWaypoint, SIGNAL(triggered()), this, SLOT(add()));

67 68 69
    // ADD WAYPOINT AT CURRENT POSITION
    connect(m_ui->positionAddButton, SIGNAL(clicked()), this, SLOT(addCurrentPositonWaypoint()));

pixhawk's avatar
pixhawk committed
70 71 72
    // SEND WAYPOINTS
    connect(m_ui->transmitButton, SIGNAL(clicked()), this, SLOT(transmit()));

73
    // REQUEST WAYPOINTS
74
    connect(m_ui->readButton, SIGNAL(clicked()), this, SLOT(read()));
75

pixhawk's avatar
pixhawk committed
76 77 78 79 80 81 82 83
    // SAVE/LOAD WAYPOINTS
    connect(m_ui->saveButton, SIGNAL(clicked()), this, SLOT(saveWaypoints()));
    connect(m_ui->loadButton, SIGNAL(clicked()), this, SLOT(loadWaypoints()));

    connect(UASManager::instance(), SIGNAL(activeUASSet(UASInterface*)), this, SLOT(setUAS(UASInterface*)));

    // SET UAS AFTER ALL SIGNALS/SLOTS ARE CONNECTED
    setUAS(uas);
84 85 86

    // STATUS LABEL
    updateStatusLabel("");
pixhawk's avatar
pixhawk committed
87 88 89 90 91 92 93
}

WaypointList::~WaypointList()
{
    delete m_ui;
}

94 95
void WaypointList::updateStatusLabel(const QString &string)
{
pixhawk's avatar
pixhawk committed
96
    m_ui->statusLabel->setText(string);
97 98
}

99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116
void WaypointList::updateLocalPosition(UASInterface* uas, double x, double y, double z, quint64 usec)
{
    Q_UNUSED(uas);
    Q_UNUSED(usec);
    mavX = x;
    mavY = y;
    mavZ = z;
}

void WaypointList::updateAttitude(UASInterface* uas, double roll, double pitch, double yaw, quint64 usec)
{
    Q_UNUSED(uas);
    Q_UNUSED(usec);
    Q_UNUSED(roll);
    Q_UNUSED(pitch);
    mavYaw = yaw;
}

pixhawk's avatar
pixhawk committed
117 118 119 120 121
void WaypointList::setUAS(UASInterface* uas)
{
    if (this->uas == NULL && uas != NULL)
    {
        this->uas = uas;
pixhawk's avatar
pixhawk committed
122

123 124 125
        connect(&uas->getWaypointManager(), SIGNAL(updateStatusString(const QString &)),                                          this, SLOT(updateStatusLabel(const QString &)));
        connect(&uas->getWaypointManager(), SIGNAL(waypointUpdated(quint16,double,double,double,double,bool,bool,double,int)),    this, SLOT(setWaypoint(quint16,double,double,double,double,bool,bool,double,int)));
        connect(&uas->getWaypointManager(), SIGNAL(currentWaypointChanged(quint16)),                                              this, SLOT(currentWaypointChanged(quint16)));
126 127
        connect(uas, SIGNAL(localPositionChanged(UASInterface*,double,double,double,quint64)), this, SLOT(updateLocalPosition(UASInterface*,double,double,double,quint64)));
        connect(uas, SIGNAL(attitudeChanged(UASInterface*,double,double,double,quint64)), this, SLOT(updateAttitude(UASInterface*,double,double,double,quint64)));
pixhawk's avatar
pixhawk committed
128

129 130 131
        connect(this, SIGNAL(sendWaypoints()),       &uas->getWaypointManager(), SLOT(sendWaypoints()));
        connect(this, SIGNAL(requestWaypoints()),   &uas->getWaypointManager(), SLOT(requestWaypoints()));
        connect(this, SIGNAL(clearWaypointList()),  &uas->getWaypointManager(), SLOT(clearWaypointList()));
pixhawk's avatar
pixhawk committed
132 133 134
    }
}

135
void WaypointList::setWaypoint(quint16 id, double x, double y, double z, double yaw, bool autocontinue, bool current, double orbit, int holdTime)
pixhawk's avatar
pixhawk committed
136
{
137
    if (this->uas)
pixhawk's avatar
pixhawk committed
138
    {
pixhawk's avatar
pixhawk committed
139
        Waypoint* wp = new Waypoint(id, x, y, z, yaw, autocontinue, current, orbit, holdTime);
pixhawk's avatar
pixhawk committed
140 141 142 143
        addWaypoint(wp);
    }
}

144
void WaypointList::waypointReached(quint16 waypointId)
pixhawk's avatar
pixhawk committed
145
{
146 147 148 149
    if (this->uas)
    {
        updateStatusLabel(QString("Waypoint %1 reached.").arg(waypointId));
    }
pixhawk's avatar
pixhawk committed
150 151 152 153
}

void WaypointList::currentWaypointChanged(quint16 seq)
{
154
    if (this->uas)
pixhawk's avatar
pixhawk committed
155
    {
156
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();
pixhawk's avatar
pixhawk committed
157

158 159 160
        if (seq < waypoints.size())
        {
            for(int i = 0; i < waypoints.size(); i++)
pixhawk's avatar
pixhawk committed
161
            {
162 163 164 165 166 167 168 169 170 171 172 173
                WaypointView* widget = wpViews.find(waypoints[i]).value();

                if (waypoints[i]->getId() == seq)
                {
                    waypoints[i]->setCurrent(true);
                    widget->setCurrent(true);
                }
                else
                {
                    waypoints[i]->setCurrent(false);
                    widget->setCurrent(false);
                }
pixhawk's avatar
pixhawk committed
174
            }
175
            redrawList();
pixhawk's avatar
pixhawk committed
176
        }
pixhawk's avatar
pixhawk committed
177
    }
pixhawk's avatar
pixhawk committed
178 179
}

180
void WaypointList::read()
pixhawk's avatar
pixhawk committed
181
{
182
    if (uas)
pixhawk's avatar
pixhawk committed
183
    {
184
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();
pixhawk's avatar
pixhawk committed
185

186 187 188 189 190 191 192
        while(waypoints.size()>0)
        {
            removeWaypoint(waypoints[0]);
        }

        emit requestWaypoints();
    }
pixhawk's avatar
pixhawk committed
193 194
}

195 196
void WaypointList::transmit()
{
197
    emit sendWaypoints();
pixhawk's avatar
pixhawk committed
198
    //emit requestWaypoints(); FIXME
199 200
}

pixhawk's avatar
pixhawk committed
201 202 203 204 205
void WaypointList::add()
{
    // Only add waypoints if UAS is present
    if (uas)
    {
206 207
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

pixhawk's avatar
pixhawk committed
208 209
        if (waypoints.size() > 0)
        {
pixhawk's avatar
pixhawk committed
210
            Waypoint *last = waypoints.at(waypoints.size()-1);
pixhawk's avatar
pixhawk committed
211
            addWaypoint(new Waypoint(waypoints.size(), last->getX(), last->getY(), last->getZ(), last->getYaw(), last->getAutoContinue(), false, last->getOrbit(), last->getHoldTime()));
pixhawk's avatar
pixhawk committed
212 213 214
        }
        else
        {
215
            addWaypoint(new Waypoint(waypoints.size(), 1.1, 1.1, -0.8, 0.0, true, true, 0.15, 2000));
pixhawk's avatar
pixhawk committed
216 217 218 219
        }
    }
}

220 221 222 223
void WaypointList::addCurrentPositonWaypoint()
{
    if (uas)
    {
224 225
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

226 227 228
        if (waypoints.size() > 0)
        {
            Waypoint *last = waypoints.at(waypoints.size()-1);
pixhawk's avatar
pixhawk committed
229
            addWaypoint(new Waypoint(waypoints.size(), (float)(qRound(mavX*100))/100.f, (float)(qRound(mavY*100))/100.f, (float)(qRound(mavZ*100))/100.f, (float)(qRound(mavYaw*100))/100.f, last->getAutoContinue(), false, last->getOrbit(), last->getHoldTime()));
230 231 232
        }
        else
        {
pixhawk's avatar
pixhawk committed
233
            addWaypoint(new Waypoint(waypoints.size(), (float)(qRound(mavX*100))/100.f, (float)(qRound(mavY*100))/100.f, (float)(qRound(mavZ*100))/100.f, (float)(qRound(mavYaw*100))/100.f, true, true, 0.15, 2000));
234 235 236 237 238
        }

    }
}

pixhawk's avatar
pixhawk committed
239 240
void WaypointList::addWaypoint(Waypoint* wp)
{
241
    if (uas)
pixhawk's avatar
pixhawk committed
242
    {
243 244 245 246 247 248 249 250 251 252 253
        uas->getWaypointManager().getWaypointList().push_back(wp);
        if (!wpViews.contains(wp))
        {
            WaypointView* wpview = new WaypointView(wp, this);
            wpViews.insert(wp, wpview);
            listLayout->addWidget(wpViews.value(wp));
            connect(wpview, SIGNAL(moveDownWaypoint(Waypoint*)), this, SLOT(moveDown(Waypoint*)));
            connect(wpview, SIGNAL(moveUpWaypoint(Waypoint*)), this, SLOT(moveUp(Waypoint*)));
            connect(wpview, SIGNAL(removeWaypoint(Waypoint*)), this, SLOT(removeWaypoint(Waypoint*)));
            connect(wpview, SIGNAL(currentWaypointChanged(quint16)), this, SLOT(currentWaypointChanged(quint16)));
        }
pixhawk's avatar
pixhawk committed
254 255 256 257 258
    }
}

void WaypointList::redrawList()
{
259
    if (uas)
pixhawk's avatar
pixhawk committed
260
    {
261 262 263 264
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

        // Clear list layout
        if (!wpViews.empty())
pixhawk's avatar
pixhawk committed
265
        {
266 267 268 269 270 271 272 273 274 275 276 277
            QMapIterator<Waypoint*,WaypointView*> viewIt(wpViews);
            viewIt.toFront();
            while(viewIt.hasNext())
            {
                viewIt.next();
                listLayout->removeWidget(viewIt.value());
            }
            // Re-add waypoints
            for(int i = 0; i < waypoints.size(); i++)
            {
                listLayout->addWidget(wpViews.value(waypoints[i]));
            }
pixhawk's avatar
pixhawk committed
278 279 280 281 282 283
        }
    }
}

void WaypointList::moveUp(Waypoint* wp)
{
284
    if (uas)
pixhawk's avatar
pixhawk committed
285
    {
286 287 288 289
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

        int id = wp->getId();
        if (waypoints.size() > 1 && waypoints.size() > id)
pixhawk's avatar
pixhawk committed
290
        {
291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306
            Waypoint* temp = waypoints[id];
            if (id > 0)
            {
                waypoints[id] = waypoints[id-1];
                waypoints[id-1] = temp;
                waypoints[id-1]->setId(id-1);
                waypoints[id]->setId(id);
            }
            else
            {
                waypoints[id] = waypoints[waypoints.size()-1];
                waypoints[waypoints.size()-1] = temp;
                waypoints[waypoints.size()-1]->setId(waypoints.size()-1);
                waypoints[id]->setId(id);
            }
            redrawList();
pixhawk's avatar
pixhawk committed
307 308 309 310 311 312
        }
    }
}

void WaypointList::moveDown(Waypoint* wp)
{
313
    if (uas)
pixhawk's avatar
pixhawk committed
314
    {
315 316 317 318
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

        int id = wp->getId();
        if (waypoints.size() > 1 && waypoints.size() > id)
pixhawk's avatar
pixhawk committed
319
        {
320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335
            Waypoint* temp = waypoints[id];
            if (id != waypoints.size()-1)
            {
                waypoints[id] = waypoints[id+1];
                waypoints[id+1] = temp;
                waypoints[id+1]->setId(id+1);
                waypoints[id]->setId(id);
            }
            else
            {
                waypoints[id] = waypoints[0];
                waypoints[0] = temp;
                waypoints[0]->setId(0);
                waypoints[id]->setId(id);
            }
            redrawList();
pixhawk's avatar
pixhawk committed
336 337 338 339 340 341
        }
    }
}

void WaypointList::removeWaypoint(Waypoint* wp)
{
342
    if (uas)
pixhawk's avatar
pixhawk committed
343
    {
344 345 346 347
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

        // Delete from list
        if (wp != NULL)
pixhawk's avatar
pixhawk committed
348
        {
349 350 351 352 353
            waypoints.remove(wp->getId());
            for(int i = wp->getId(); i < waypoints.size(); i++)
            {
                waypoints[i]->setId(i);
            }
pixhawk's avatar
pixhawk committed
354

355 356 357 358 359 360 361
            // Remove from view
            WaypointView* widget = wpViews.find(wp).value();
            wpViews.remove(wp);
            widget->hide();
            listLayout->removeWidget(widget);
            delete wp;
        }
pixhawk's avatar
pixhawk committed
362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377
    }
}

void WaypointList::changeEvent(QEvent *e)
{
    switch (e->type()) {
    case QEvent::LanguageChange:
        m_ui->retranslateUi(this);
        break;
    default:
        break;
    }
}

void WaypointList::saveWaypoints()
{
378
    if (uas)
pixhawk's avatar
pixhawk committed
379
    {
380 381 382 383 384 385 386 387 388 389 390 391 392 393 394
        QString fileName = QFileDialog::getSaveFileName(this, tr("Save File"), "./waypoints.txt", tr("Waypoint File (*.txt)"));
        QFile file(fileName);
        if (!file.open(QIODevice::WriteOnly | QIODevice::Text))
            return;

        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

        QTextStream in(&file);
        for (int i = 0; i < waypoints.size(); i++)
        {
            Waypoint* wp = waypoints[i];
            in << "\t" << wp->getId() << "\t" << wp->getX() << "\t" << wp->getY()  << "\t" << wp->getZ()  << "\t" << wp->getYaw()  << "\t" << wp->getAutoContinue() << "\t" << wp->getCurrent() << "\t" << wp->getOrbit() << "\t" << wp->getHoldTime() << "\n";
            in.flush();
        }
        file.close();
pixhawk's avatar
pixhawk committed
395 396 397 398 399
    }
}

void WaypointList::loadWaypoints()
{
400
    if (uas)
pixhawk's avatar
pixhawk committed
401
    {
402 403 404 405
        QString fileName = QFileDialog::getOpenFileName(this, tr("Load File"), ".", tr("Waypoint File (*.txt)"));
        QFile file(fileName);
        if (!file.open(QIODevice::ReadOnly | QIODevice::Text))
            return;
pixhawk's avatar
pixhawk committed
406

407 408 409 410 411 412 413 414 415 416 417 418 419 420 421
        QVector<Waypoint *> &waypoints = uas->getWaypointManager().getWaypointList();

        while(waypoints.size()>0)
        {
            removeWaypoint(waypoints[0]);
        }

        QTextStream in(&file);
        while (!in.atEnd())
        {
            QStringList wpParams = in.readLine().split("\t");
            if (wpParams.size() == 10)
                addWaypoint(new Waypoint(wpParams[1].toInt(), wpParams[2].toDouble(), wpParams[3].toDouble(), wpParams[4].toDouble(), wpParams[5].toDouble(), (wpParams[6].toInt() == 1 ? true : false), (wpParams[7].toInt() == 1 ? true : false), wpParams[8].toDouble(), wpParams[9].toInt()));
        }
        file.close();
pixhawk's avatar
pixhawk committed
422 423 424
    }
}