/*===================================================================== PIXHAWK Micro Air Vehicle Flying Robotics Toolkit (c) 2009, 2010 PIXHAWK PROJECT 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 . ======================================================================*/ /** * @file * @brief Waypoint list widget * * @author Lorenz Meier * @author Benjamin Knecht * @author Petri Tanskanen * */ #include "WaypointList.h" #include "ui_WaypointList.h" #include #include #include #include WaypointList::WaypointList(QWidget *parent, UASInterface* uas) : QWidget(parent), uas(NULL), mavX(0.0), mavY(0.0), mavZ(0.0), mavYaw(0.0), 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(); 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())); // ADD WAYPOINT AT CURRENT POSITION connect(m_ui->positionAddButton, SIGNAL(clicked()), this, SLOT(addCurrentPositonWaypoint())); // SEND WAYPOINTS connect(m_ui->transmitButton, SIGNAL(clicked()), this, SLOT(transmit())); // REQUEST WAYPOINTS connect(m_ui->readButton, SIGNAL(clicked()), this, SLOT(read())); // 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); // STATUS LABEL updateStatusLabel(""); } WaypointList::~WaypointList() { delete m_ui; } void WaypointList::updateStatusLabel(const QString &string) { m_ui->statusLabel->setText(string); } 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; } void WaypointList::setUAS(UASInterface* uas) { if (this->uas == NULL && uas != NULL) { this->uas = uas; 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))); 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))); connect(this, SIGNAL(sendWaypoints()), &uas->getWaypointManager(), SLOT(writeWaypoints())); connect(this, SIGNAL(requestWaypoints()), &uas->getWaypointManager(), SLOT(readWaypoints())); connect(this, SIGNAL(clearWaypointList()), &uas->getWaypointManager(), SLOT(clearWaypointList())); connect(this, SIGNAL(setCurrent(quint16)), &uas->getWaypointManager(), SLOT(setCurrent(quint16))); } } void WaypointList::setWaypoint(quint16 id, double x, double y, double z, double yaw, bool autocontinue, bool current, double orbit, int holdTime) { if (this->uas) { Waypoint* wp = new Waypoint(id, x, y, z, yaw, autocontinue, current, orbit, holdTime); addWaypoint(wp); } } void WaypointList::waypointReached(quint16 waypointId) { if (this->uas) { updateStatusLabel(QString("Waypoint %1 reached.").arg(waypointId)); } } void WaypointList::changeCurrentWaypoint(quint16 seq) { if (this->uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); if (seq < waypoints.size()) { for(int i = 0; i < waypoints.size(); i++) { WaypointView* widget = wpViews.find(waypoints[i]).value(); if (waypoints[i]->getId() == seq) { waypoints[i]->setCurrent(true); widget->setCurrent(true); emit setCurrent(seq); } else { waypoints[i]->setCurrent(false); widget->setCurrent(false); } } redrawList(); } } } void WaypointList::currentWaypointChanged(quint16 seq) { if (this->uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); if (seq < waypoints.size()) { for(int i = 0; i < waypoints.size(); i++) { 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); } } redrawList(); } } } void WaypointList::read() { if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); while(waypoints.size()>0) { removeWaypoint(waypoints[0]); } emit requestWaypoints(); } } void WaypointList::transmit() { emit sendWaypoints(); //emit requestWaypoints(); FIXME } void WaypointList::add() { // Only add waypoints if UAS is present if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); if (waypoints.size() > 0) { Waypoint *last = waypoints.at(waypoints.size()-1); addWaypoint(new Waypoint(waypoints.size(), last->getX(), last->getY(), last->getZ(), last->getYaw(), last->getAutoContinue(), false, last->getOrbit(), last->getHoldTime())); } else { addWaypoint(new Waypoint(waypoints.size(), 1.1, 1.1, -0.8, 0.0, true, true, 0.15, 2000)); } } } void WaypointList::addCurrentPositonWaypoint() { if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); if (waypoints.size() > 0) { Waypoint *last = waypoints.at(waypoints.size()-1); 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())); } else { 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)); } } } void WaypointList::addWaypoint(Waypoint* wp) { if (uas) { 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))); connect(wpview, SIGNAL(changeCurrentWaypoint(quint16)), this, SLOT(changeCurrentWaypoint(quint16))); } } } void WaypointList::redrawList() { if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); // Clear list layout if (!wpViews.empty()) { QMapIterator 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])); } } } } void WaypointList::moveUp(Waypoint* wp) { if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); int id = wp->getId(); if (waypoints.size() > 1 && waypoints.size() > id) { 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(); } } } void WaypointList::moveDown(Waypoint* wp) { if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); int id = wp->getId(); if (waypoints.size() > 1 && waypoints.size() > id) { 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(); } } } void WaypointList::removeWaypoint(Waypoint* wp) { if (uas) { QVector &waypoints = uas->getWaypointManager().getWaypointList(); // Delete from list if (wp != NULL) { waypoints.remove(wp->getId()); for(int i = wp->getId(); i < waypoints.size(); i++) { waypoints[i]->setId(i); } // Remove from view WaypointView* widget = wpViews.find(wp).value(); wpViews.remove(wp); widget->hide(); listLayout->removeWidget(widget); delete wp; } } } void WaypointList::changeEvent(QEvent *e) { switch (e->type()) { case QEvent::LanguageChange: m_ui->retranslateUi(this); break; default: break; } } void WaypointList::saveWaypoints() { if (uas) { 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 &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(); } } void WaypointList::loadWaypoints() { if (uas) { QString fileName = QFileDialog::getOpenFileName(this, tr("Load File"), ".", tr("Waypoint File (*.txt)")); QFile file(fileName); if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) return; QVector &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(); } }