/*===================================================================== 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 Implementation of one airstrip * * @author Lorenz Meier <mavteam@student.ethz.ch> * */ #include <cmath> #include <QDateTime> #include <QDebug> #include <QMenu> #include <QInputDialog> #include "QGC.h" #include "UASManager.h" #include "UASView.h" #include "UASWaypointManager.h" #include "MainWindow.h" #include "ui_UASView.h" UASView::UASView(UASInterface* uas, QWidget *parent) : QWidget(parent), startTime(0), timeout(false), iconIsRed(true), timeRemaining(0), chargeLevel(0), uas(uas), load(0), state("UNKNOWN"), stateDesc(tr("Unknown state")), mode("MAV_MODE_UNKNOWN"), thrust(0), isActive(false), x(0), y(0), z(0), totalSpeed(0), lat(0), lon(0), alt(0), groundDistance(0), localFrame(false), globalFrameKnown(false), removeAction(new QAction("Delete this system", this)), renameAction(new QAction("Rename..", this)), selectAction(new QAction("Control this system", this )), hilAction(new QAction("Enable Hardware-in-the-Loop Simulation", this )), selectAirframeAction(new QAction("Choose Airframe", this)), setBatterySpecsAction(new QAction("Set Battery Options", this)), lowPowerModeEnabled(true), generalUpdateCount(0), filterTime(0), m_ui(new Ui::UASView) { // FIXME XXX lowPowerModeEnabled = MainWindow::instance()->lowPowerModeEnabled(); hilAction->setCheckable(true); m_ui->setupUi(this); // Setup communication //connect(uas, SIGNAL(valueChanged(int,QString,double,quint64)), this, SLOT(receiveValue(int,QString,double,quint64))); connect(uas, SIGNAL(batteryChanged(UASInterface*, double, double, int)), this, SLOT(updateBattery(UASInterface*, double, double, int))); connect(uas, SIGNAL(heartbeat(UASInterface*)), this, SLOT(receiveHeartbeat(UASInterface*))); connect(uas, SIGNAL(thrustChanged(UASInterface*, double)), this, SLOT(updateThrust(UASInterface*, double))); connect(uas, SIGNAL(localPositionChanged(UASInterface*,double,double,double,quint64)), this, SLOT(updateLocalPosition(UASInterface*,double,double,double,quint64))); connect(uas, SIGNAL(globalPositionChanged(UASInterface*,double,double,double,quint64)), this, SLOT(updateGlobalPosition(UASInterface*,double,double,double,quint64))); connect(uas, SIGNAL(speedChanged(UASInterface*,double,double,double,quint64)), this, SLOT(updateSpeed(UASInterface*,double,double,double,quint64))); connect(uas, SIGNAL(statusChanged(UASInterface*,QString,QString)), this, SLOT(updateState(UASInterface*,QString,QString))); connect(uas, SIGNAL(modeChanged(int,QString,QString)), this, SLOT(updateMode(int,QString,QString))); connect(uas, SIGNAL(loadChanged(UASInterface*, double)), this, SLOT(updateLoad(UASInterface*, double))); connect(uas, SIGNAL(heartbeatTimeout()), this, SLOT(heartbeatTimeout())); connect(uas, SIGNAL(waypointSelected(int,int)), this, SLOT(selectWaypoint(int,int))); connect(uas->getWaypointManager(), SIGNAL(currentWaypointChanged(quint16)), this, SLOT(currentWaypointUpdated(quint16))); connect(uas, SIGNAL(systemTypeSet(UASInterface*,uint)), this, SLOT(setSystemType(UASInterface*,uint))); connect(UASManager::instance(), SIGNAL(activeUASStatusChanged(UASInterface*,bool)), this, SLOT(updateActiveUAS(UASInterface*,bool))); connect(uas, SIGNAL(textMessageReceived(int,int,int,QString)), this, SLOT(showStatusText(int, int, int, QString))); connect(uas, SIGNAL(navModeChanged(int, int, QString)), this, SLOT(updateNavMode(int, int, QString))); // Setup UAS selection connect(m_ui->uasViewFrame, SIGNAL(clicked(bool)), this, SLOT(setUASasActive(bool))); // Setup user interaction connect(m_ui->liftoffButton, SIGNAL(clicked()), uas, SLOT(launch())); connect(m_ui->haltButton, SIGNAL(clicked()), uas, SLOT(halt())); connect(m_ui->continueButton, SIGNAL(clicked()), uas, SLOT(go())); connect(m_ui->landButton, SIGNAL(clicked()), uas, SLOT(home())); connect(m_ui->abortButton, SIGNAL(clicked()), uas, SLOT(emergencySTOP())); connect(m_ui->killButton, SIGNAL(clicked()), uas, SLOT(emergencyKILL())); connect(m_ui->shutdownButton, SIGNAL(clicked()), uas, SLOT(shutdown())); // Allow to delete this widget connect(removeAction, SIGNAL(triggered()), this, SLOT(deleteLater())); connect(renameAction, SIGNAL(triggered()), this, SLOT(rename())); connect(selectAction, SIGNAL(triggered()), uas, SLOT(setSelected())); connect(hilAction, SIGNAL(triggered(bool)), uas, SLOT(enableHil(bool))); connect(selectAirframeAction, SIGNAL(triggered()), this, SLOT(selectAirframe())); connect(setBatterySpecsAction, SIGNAL(triggered()), this, SLOT(setBatterySpecs())); connect(uas, SIGNAL(systemRemoved()), this, SLOT(deleteLater())); // Name changes connect(uas, SIGNAL(nameChanged(QString)), this, SLOT(updateName(QString))); // Set static values // Name if (uas->getUASName() == "") { m_ui->nameLabel->setText(tr("UAS") + QString::number(uas->getUASID())); } else { m_ui->nameLabel->setText(uas->getUASName()); } setBackgroundColor(); // Heartbeat fade refreshTimer = new QTimer(this); connect(refreshTimer, SIGNAL(timeout()), this, SLOT(refresh())); if (lowPowerModeEnabled) { refreshTimer->start(updateInterval*3); } else { refreshTimer->start(updateInterval); } // Hide kill and shutdown buttons per default m_ui->killButton->hide(); m_ui->shutdownButton->hide(); // Set state and mode updateMode(uas->getUASID(), uas->getShortMode(), ""); updateState(uas, uas->getShortState(), ""); setSystemType(uas, uas->getSystemType()); } UASView::~UASView() { delete m_ui; delete removeAction; delete renameAction; delete selectAction; } void UASView::heartbeatTimeout() { timeout = true; } void UASView::updateNavMode(int uasid, int mode, const QString& text) { Q_UNUSED(uasid); Q_UNUSED(mode); m_ui->navLabel->setText(text); } void UASView::showStatusText(int uasid, int componentid, int severity, QString text) { Q_UNUSED(uasid); Q_UNUSED(componentid); Q_UNUSED(severity); //m_ui->statusTextLabel->setText(text); stateDesc = text; } /** * Set the background color based on the MAV color. If the MAV is selected as the * currently actively controlled system, the frame color is highlighted */ void UASView::setBackgroundColor() { // UAS color QColor uasColor = uas->getColor(); QString colorstyle; QString borderColor = "#4A4A4F"; if (isActive) { borderColor = "#FA4A4F"; uasColor = uasColor.darker(475); } else { uasColor = uasColor.darker(675); } colorstyle = colorstyle.sprintf("QGroupBox { border-radius: 12px; padding: 0px; margin: 0px; background-color: #%02X%02X%02X; border: 2px solid %s; }", uasColor.red(), uasColor.green(), uasColor.blue(), borderColor.toStdString().c_str()); m_ui->uasViewFrame->setStyleSheet(colorstyle); } void UASView::setUASasActive(bool active) { if (active) { UASManager::instance()->setActiveUAS(this->uas); } } void UASView::updateActiveUAS(UASInterface* uas, bool active) { if (uas == this->uas) { this->isActive = active; setBackgroundColor(); } } void UASView::updateMode(int sysId, QString status, QString description) { Q_UNUSED(description); //int aa=this->uas->getUASID(); if (sysId == this->uas->getUASID()) m_ui->modeLabel->setText(status); m_ui->modeLabel->setText(status); } void UASView::mouseDoubleClickEvent (QMouseEvent * event) { Q_UNUSED(event); UASManager::instance()->setActiveUAS(uas); // qDebug() << __FILE__ << __LINE__ << "DOUBLECLICKED"; } void UASView::enterEvent(QEvent* event) { if (event->type() == QEvent::MouseMove) { emit uasInFocus(uas); if (uas != UASManager::instance()->getActiveUAS()) { grabMouse(QCursor(Qt::PointingHandCursor)); } } if (event->type() == QEvent::MouseButtonDblClick) { // qDebug() << __FILE__ << __LINE__ << "UAS CLICKED!"; } } void UASView::leaveEvent(QEvent* event) { if (event->type() == QEvent::MouseMove) { emit uasOutFocus(uas); releaseMouse(); } } void UASView::showEvent(QShowEvent* event) { // React only to internal (pre-display) // events Q_UNUSED(event); refreshTimer->start(updateInterval*10); } void UASView::hideEvent(QHideEvent* event) { // React only to internal (pre-display) // events Q_UNUSED(event); refreshTimer->stop(); } void UASView::receiveHeartbeat(UASInterface* uas) { Q_UNUSED(uas); heartbeatColor = QColor(20, 200, 20); QString colorstyle("QGroupBox { border-radius: 5px; padding: 2px; margin: 0px; border: 0px; background-color: %1; }"); m_ui->heartbeatIcon->setStyleSheet(colorstyle.arg(heartbeatColor.name())); if (timeout) setBackgroundColor(); timeout = false; } void UASView::updateName(const QString& name) { if (uas) m_ui->nameLabel->setText(name); } /** * The current system type is represented through the system icon. * * @param uas Source system, has to be the same as this->uas * @param systemType type ID, following the MAVLink system type conventions * @see http://pixhawk.ethz.ch/software/mavlink */ void UASView::setSystemType(UASInterface* uas, unsigned int systemType) { if (uas == this->uas) { // Set matching icon switch (systemType) { case 0: m_ui->typeButton->setIcon(QIcon(":/images/mavs/generic.svg")); break; case 1: m_ui->typeButton->setIcon(QIcon(":/images/mavs/fixed-wing.svg")); break; case 2: m_ui->typeButton->setIcon(QIcon(":/images/mavs/quadrotor.svg")); break; case 3: m_ui->typeButton->setIcon(QIcon(":/images/mavs/coaxial.svg")); break; case 4: m_ui->typeButton->setIcon(QIcon(":/images/mavs/helicopter.svg")); break; case 5: m_ui->typeButton->setIcon(QIcon(":/images/mavs/unknown.svg")); break; case 6: { // A groundstation is a special system type, update widget QString result; m_ui->nameLabel->setText(tr("GCS ") + result.sprintf("%03d", uas->getUASID())); m_ui->waypointLabel->setText(""); m_ui->timeRemainingLabel->setText("Online:"); m_ui->batteryBar->hide(); m_ui->thrustBar->hide(); m_ui->stateLabel->hide(); m_ui->statusTextLabel->hide(); m_ui->waypointLabel->hide(); m_ui->liftoffButton->hide(); m_ui->haltButton->hide(); m_ui->landButton->hide(); m_ui->shutdownButton->hide(); m_ui->abortButton->hide(); m_ui->typeButton->setIcon(QIcon(":/images/mavs/groundstation.svg")); } break; default: m_ui->typeButton->setIcon(QIcon(":/images/mavs/unknown.svg")); break; } } } void UASView::updateLocalPosition(UASInterface* uas, double x, double y, double z, quint64 usec) { Q_UNUSED(usec); Q_UNUSED(uas); this->x = x; this->y = y; this->z = z; localFrame = true; } void UASView::updateGlobalPosition(UASInterface* uas, double lon, double lat, double alt, quint64 usec) { Q_UNUSED(uas); Q_UNUSED(usec); this->lon = lon; this->lat = lat; this->alt = alt; globalFrameKnown = true; } void UASView::updateSpeed(UASInterface*, double x, double y, double z, quint64 usec) { Q_UNUSED(usec); totalSpeed = sqrt(x*x + y*y + z*z); } void UASView::currentWaypointUpdated(quint16 waypoint) { m_ui->waypointLabel->setText(tr("WP") + QString::number(waypoint)); } void UASView::setWaypoint(int uasId, int id, double x, double y, double z, double yaw, bool autocontinue, bool current) { Q_UNUSED(x); Q_UNUSED(y); Q_UNUSED(z); Q_UNUSED(yaw); Q_UNUSED(autocontinue); if (uasId == this->uas->getUASID()) { if (current) { m_ui->waypointLabel->setText(tr("WP") + QString::number(id)); } } } void UASView::selectWaypoint(int uasId, int id) { if (uasId == this->uas->getUASID()) { m_ui->waypointLabel->setText(tr("WP") + QString::number(id)); } } void UASView::updateThrust(UASInterface* uas, double thrust) { if (this->uas == uas) { this->thrust = thrust; } } void UASView::updateBattery(UASInterface* uas, double voltage, double percent, int seconds) { Q_UNUSED(voltage); if (this->uas == uas) { timeRemaining = seconds; chargeLevel = percent; } } void UASView::updateState(UASInterface* uas, QString uasState, QString stateDescription) { if (this->uas == uas) { state = uasState; stateDesc = stateDescription; } } void UASView::updateLoad(UASInterface* uas, double load) { if (this->uas == uas) { this->load = load; } } void UASView::contextMenuEvent (QContextMenuEvent* event) { QMenu menu(this); menu.addAction(selectAction); menu.addSeparator(); menu.addAction(renameAction); if (timeout) { menu.addAction(removeAction); } menu.addAction(hilAction); menu.addAction(selectAirframeAction); menu.addAction(setBatterySpecsAction); menu.exec(event->globalPos()); } void UASView::setBatterySpecs() { if (uas) { bool ok; QString newName = QInputDialog::getText(this, tr("Set Battery Specifications for %1").arg(uas->getUASName()), tr("Specs: (empty,warn,full), e.g. (9V,9.5V,12.6V) or just warn level in percent (e.g. 15%) to use estimate from MAV"), QLineEdit::Normal, uas->getBatterySpecs(), &ok); if (ok && !newName.isEmpty()) uas->setBatterySpecs(newName); } } void UASView::rename() { if (uas) { bool ok; QString newName = QInputDialog::getText(this, tr("Rename System %1").arg(uas->getUASName()), tr("System Name:"), QLineEdit::Normal, uas->getUASName(), &ok); if (ok && !newName.isEmpty()) uas->setUASName(newName); } } void UASView::selectAirframe() { if (uas) { // Get list of airframes from UAS QStringList airframes; airframes << "Generic" << "Multiplex Easystar" << "Multiplex Twinstar" << "Multiplex Merlin" << "Pixhawk Cheetah" << "Mikrokopter" << "Reaper" << "Predator" << "Coaxial" << "Pteryx" << "Asctec Firefly"; bool ok; QString item = QInputDialog::getItem(this, tr("Select Airframe for %1").arg(uas->getUASName()), tr("Airframe"), airframes, uas->getAirframe(), false, &ok); if (ok && !item.isEmpty()) { // Set this airframe as UAS airframe uas->setAirframe(airframes.indexOf(item)); } } } void UASView::refresh() { //setUpdatesEnabled(false); //setUpdatesEnabled(true); //repaint(); //qDebug() << "UPDATING UAS WIDGET!" << uas->getUASName(); quint64 lastupdate = 0; //// qDebug() << "UASVIEW update diff: " << MG::TIME::getGroundTimeNow() - lastupdate; lastupdate = QGC::groundTimeMilliseconds(); if (generalUpdateCount == 4) { #if (QGC_EVENTLOOP_DEBUG) // qDebug() << "EVENTLOOP:" << __FILE__ << __LINE__; #endif generalUpdateCount = 0; //// qDebug() << "UPDATING EVERYTHING"; // State m_ui->stateLabel->setText(state); m_ui->statusTextLabel->setText(stateDesc); // Battery m_ui->batteryBar->setValue(static_cast<int>(this->chargeLevel)); //m_ui->loadBar->setValue(static_cast<int>(this->load)); m_ui->thrustBar->setValue(this->thrust); // Position // If global position is known, prefer it over local coordinates if (!globalFrameKnown && localFrame) { QString position; position = position.sprintf("%05.1f %05.1f %06.1f m", x, y, z); m_ui->positionLabel->setText(position); } if (globalFrameKnown) { QString globalPosition; QString latIndicator; if (lat > 0) { latIndicator = "N"; } else { latIndicator = "S"; } QString lonIndicator; if (lon > 0) { lonIndicator = "E"; } else { lonIndicator = "W"; } globalPosition = globalPosition.sprintf("%05.1f%s %05.1f%s %06.1f m", lon, lonIndicator.toStdString().c_str(), lat, latIndicator.toStdString().c_str(), alt); m_ui->positionLabel->setText(globalPosition); } // Altitude if (groundDistance == 0 && alt != 0) { m_ui->groundDistanceLabel->setText(QString("%1 m").arg(alt, 6, 'f', 1, '0')); } else { m_ui->groundDistanceLabel->setText(QString("%1 m").arg(groundDistance, 6, 'f', 1, '0')); } // Speed QString speed("%1 m/s"); m_ui->speedLabel->setText(speed.arg(totalSpeed, 4, 'f', 1, '0')); // Thrust m_ui->thrustBar->setValue(thrust * 100); if(this->timeRemaining > 1 && this->timeRemaining < QGC::MAX_FLIGHT_TIME) { // Filter output to get a higher stability filterTime = static_cast<int>(this->timeRemaining); filterTime = 0.8 * filterTime + 0.2 * static_cast<int>(this->timeRemaining); int sec = static_cast<int>(filterTime - static_cast<int>(filterTime / 60.0f) * 60); int min = static_cast<int>(filterTime / 60); int hours = static_cast<int>(filterTime - min * 60 - sec); QString timeText; timeText = timeText.sprintf("%02d:%02d:%02d", hours, min, sec); m_ui->timeRemainingLabel->setText(timeText); } else { m_ui->timeRemainingLabel->setText(tr("Calc..")); } // Time Elapsed //QDateTime time = MG::TIME::msecToQDateTime(uas->getUptime()); quint64 filterTime = uas->getUptime() / 1000; int sec = static_cast<int>(filterTime - static_cast<int>(filterTime / 60) * 60); int min = static_cast<int>(filterTime / 60); int hours = static_cast<int>(filterTime - min * 60 - sec); QString timeText; timeText = timeText.sprintf("%02d:%02d:%02d", hours, min, sec); m_ui->timeElapsedLabel->setText(timeText); } generalUpdateCount++; QString colorstyle("QGroupBox { border-radius: 5px; padding: 2px; margin: 0px; border: 0px; background-color: %1; }"); if (timeout) { // CRITICAL CONDITION, NO HEARTBEAT QString borderColor = "#FFFF00"; if (isActive) { borderColor = "#FA4A4F"; } if (iconIsRed) { QColor warnColor(Qt::red); m_ui->heartbeatIcon->setStyleSheet(colorstyle.arg(warnColor.name())); QString style = QString("QGroupBox { border-radius: 12px; padding: 0px; margin: 0px; border: 2px solid %1; background-color: %2; }").arg(borderColor, warnColor.name()); m_ui->uasViewFrame->setStyleSheet(style); } else { QColor warnColor(Qt::black); m_ui->heartbeatIcon->setStyleSheet(colorstyle.arg(warnColor.name())); QString style = QString("QGroupBox { border-radius: 12px; padding: 0px; margin: 0px; border: 2px solid %1; background-color: %2; }").arg(borderColor, warnColor.name()); m_ui->uasViewFrame->setStyleSheet(style); refreshTimer->setInterval(errorUpdateInterval); refreshTimer->start(); } iconIsRed = !iconIsRed; } else { if (!lowPowerModeEnabled) { // Fade heartbeat icon // Make color darker heartbeatColor = heartbeatColor.darker(210); //m_ui->heartbeatIcon->setAutoFillBackground(true); m_ui->heartbeatIcon->setStyleSheet(colorstyle.arg(heartbeatColor.name())); refreshTimer->setInterval(updateInterval); refreshTimer->start(); } } //setUpdatesEnabled(true); //setUpdatesEnabled(false); } void UASView::changeEvent(QEvent *e) { QWidget::changeEvent(e); switch (e->type()) { case QEvent::LanguageChange: m_ui->retranslateUi(this); break; default: break; } }