/*===================================================================== 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 Implementation of Horizontal Situation Indicator class * * @author Lorenz Meier * */ #include #include #include #include "UASManager.h" #include "HSIDisplay.h" #include "MG.h" #include HSIDisplay::HSIDisplay(QWidget *parent) : HDDisplay(NULL, parent), gpsSatellites(), satellitesUsed(0) { connect(UASManager::instance(), SIGNAL(activeUASSet(UASInterface*)), this, SLOT(setActiveUAS(UASInterface*))); } void HSIDisplay::paintEvent(QPaintEvent * event) { Q_UNUSED(event); //paintGL(); static quint64 interval = 0; qDebug() << "INTERVAL:" << MG::TIME::getGroundTimeNow() - interval << __FILE__ << __LINE__; interval = MG::TIME::getGroundTimeNow(); paintDisplay(); } void HSIDisplay::paintDisplay() { quint64 refreshInterval = 100; quint64 currTime = MG::TIME::getGroundTimeNow(); if (currTime - lastPaintTime < refreshInterval) { // FIXME Need to find the source of the spurious paint events //return; } lastPaintTime = currTime; // Draw instruments // TESTING THIS SHOULD BE MOVED INTO A QGRAPHICSVIEW // Update scaling factor // adjust scaling to fit both horizontally and vertically scalingFactor = this->width()/vwidth; double scalingFactorH = this->height()/vheight; if (scalingFactorH < scalingFactor) scalingFactor = scalingFactorH; QPainter painter(this); painter.setRenderHint(QPainter::Antialiasing, true); painter.setRenderHint(QPainter::HighQualityAntialiasing, true); painter.fillRect(QRect(0, 0, width(), height()), backgroundColor); const int columns = 3; const float spacing = 0.4f; // 40% of width const float gaugeWidth = vwidth / (((float)columns) + (((float)columns+1) * spacing + spacing * 0.1f)); const QColor ringColor = QColor(200, 250, 200); const int ringCount = 2; const float margin = 0.1f; // 10% margin of total width on each side for (int i = 0; i < ringCount; i++) { float radius = (vwidth - vwidth * 2.0f * margin) / (2.0f * i+1) / 2.0f; drawCircle(vwidth/2.0f, vheight/2.0f, radius, 0.1f, ringColor, &painter); } // Draw center indicator drawCircle(vwidth/2.0f, vheight/2.0f, 1.0f, 0.1f, ringColor, &painter); drawGPS(); // Draw state indicator // Draw position //drawSystemIndicator(10.0f-gaugeWidth/2.0f, 20.0f, 10.0f, 40.0f, 15.0f, &painter); //drawGauge(15.0f, 15.0f, gaugeWidth/2.0f, 0, 1.0f, "thrust", values.value("thrust", 0.0f), gaugeColor, &painter, qMakePair(0.45f, 0.8f), qMakePair(0.8f, 1.0f), true); //drawGauge(15.0f+gaugeWidth*1.7f, 15.0f, gaugeWidth/2.0f, 0, 10.0f, "altitude", values.value("altitude", 0.0f), gaugeColor, &painter, qMakePair(1.0f, 2.5f), qMakePair(0.0f, 0.5f), true); // Left spacing from border / other gauges, measured from left edge to center // float leftSpacing = gaugeWidth * spacing; // float xCoord = leftSpacing + gaugeWidth/2.0f; // // float topSpacing = leftSpacing; // float yCoord = topSpacing + gaugeWidth/2.0f; // // for (int i = 0; i < acceptList->size(); ++i) // { // QString value = acceptList->at(i); // drawGauge(xCoord, yCoord, gaugeWidth/2.0f, minValues.value(value, -1.0f), maxValues.value(value, 1.0f), value, values.value(value, minValues.value(value, 0.0f)), gaugeColor, &painter, goodRanges.value(value, qMakePair(0.0f, 0.5f)), critRanges.value(value, qMakePair(0.7f, 1.0f)), true); // xCoord += gaugeWidth + leftSpacing; // // Move one row down if necessary // if (xCoord + gaugeWidth > vwidth) // { // yCoord += topSpacing + gaugeWidth; // xCoord = leftSpacing + gaugeWidth/2.0f; // } // } } /** * * @param uas the UAS/MAV to monitor/display with the HUD */ void HSIDisplay::setActiveUAS(UASInterface* uas) { HDDisplay::setActiveUAS(uas); //qDebug() << "ATTEMPTING TO SET UAS"; if (this->uas != NULL && this->uas != uas) { // Disconnect any previously connected active MAV //disconnect(uas, SIGNAL(valueChanged(UASInterface*,QString,double,quint64)), this, SLOT(updateValue(UASInterface*,QString,double,quint64))); } connect(uas, SIGNAL(gpsSatelliteStatusChanged(int,int,float,float,float,bool)), this, SLOT(updateSatellite(int,int,float,float,float,bool))); connect(uas, SIGNAL(localPositionChanged(UASInterface*,double,double,double,quint64)), this, SLOT()) // Now connect the new UAS //if (this->uas != uas) // { //qDebug() << "UAS SET!" << "ID:" << uas->getUASID(); // Setup communication //connect(uas, SIGNAL(valueChanged(UASInterface*,QString,double,quint64)), this, SLOT(updateValue(UASInterface*,QString,double,quint64))); //} } void HSIDisplay::updateAttitudeSetpoints(UASInterface*, double rollDesired, double pitchDesired, double yawDesired, double thrustDesired, quint64 usec) { } void HSIDisplay::updatePositionSetpoints(UASInterface*, double rollDesired, double pitchDesired, double yawDesired, double thrustDesired, quint64 usec) { } void HSIDisplay::updateLocalPosition(UASInterface*, double x, double y, double z, quint64 usec) { } void HSIDisplay::updateGlobalPosition(UASInterface*, double lat, double lon, double alt, quint64 usec) { } void HSIDisplay::updateSatellite(int uasid, int satid, float elevation, float azimuth, float snr, bool used) { Q_UNUSED(uasid); //qDebug() << "UPDATED SATELLITE"; // If slot is empty, insert object if (gpsSatellites.contains(satid)) { gpsSatellites.value(satid)->update(satid, elevation, azimuth, snr, used); } else { gpsSatellites.insert(satid, new GPSSatellite(satid, elevation, azimuth, snr, used)); } } QColor HSIDisplay::getColorForSNR(float snr) { QColor color; if (snr > 0 && snr < 30) { color = QColor(250, 10, 10); } else if (snr >= 30 && snr < 35) { color = QColor(230, 230, 10); } else if (snr >= 35 && snr < 40) { color = QColor(90, 200, 90); } else if (snr >= 40) { color = QColor(20, 200, 20); } else { color = QColor(180, 180, 180); } return color; } void HSIDisplay::drawGPS() { float xCenter = vwidth/2.0f; float yCenter = vwidth/2.0f; QPainter painter(this); painter.setRenderHint(QPainter::Antialiasing, true); painter.setRenderHint(QPainter::HighQualityAntialiasing, true); // Max satellite circle radius const float margin = 0.15f; // 20% margin of total width on each side float radius = (vwidth - vwidth * 2.0f * margin) / 2.0f; quint64 currTime = MG::TIME::getGroundTimeNowUsecs(); // Draw satellite labels // QString label; // label.sprintf("%05.1f", value); // paintText(label, color, 4.5f, xRef-7.5f, yRef-2.0f, painter); QMapIterator i(gpsSatellites); while (i.hasNext()) { i.next(); GPSSatellite* sat = i.value(); // Check if update is not older than 5 seconds, else delete satellite if (sat->lastUpdate + 1000000 < currTime) { // Delete and go to next satellite gpsSatellites.remove(i.key()); if (i.hasNext()) { i.next(); sat = i.value(); } else { continue; } } if (sat) { // Draw satellite QBrush brush; QColor color = getColorForSNR(sat->snr); brush.setColor(color); if (sat->used) { brush.setStyle(Qt::SolidPattern); } else { brush.setStyle(Qt::NoBrush); } painter.setPen(Qt::SolidLine); painter.setPen(color); painter.setBrush(brush); float xPos = xCenter + (sin(((sat->azimuth/255.0f)*360.0f)/180.0f * M_PI) * cos(sat->elevation/180.0f * M_PI)) * radius; float yPos = yCenter - (cos(((sat->azimuth/255.0f)*360.0f)/180.0f * M_PI) * cos(sat->elevation/180.0f * M_PI)) * radius; drawCircle(xPos, yPos, vwidth*0.02f, 1.0f, color, &painter); paintText(QString::number(sat->id), QColor(255, 255, 255), 2.9f, xPos+1.7f, yPos+2.0f, &painter); } } } void HSIDisplay::drawObjects() { } void HSIDisplay::drawBaseLines(float xRef, float yRef, float radius, float yaw, const QColor& color, QPainter* painter, bool solid) { // Draw the circle QPen circlePen(Qt::SolidLine); if (!solid) circlePen.setStyle(Qt::DotLine); circlePen.setWidth(refLineWidthToPen(0.5f)); circlePen.setColor(defaultColor); painter->setBrush(Qt::NoBrush); painter->setPen(circlePen); drawCircle(xRef, yRef, radius, 200.0f, color, painter); //drawCircle(xRef, yRef, radius, 200.0f, 170.0f, 1.0f, color, painter); // // Draw the value // QString label; // label.sprintf("%05.1f", value); // paintText(label, color, 4.5f, xRef-7.5f, yRef-2.0f, painter); // Draw the needle // Scale the rotation so that the gauge does one revolution // per max. change const float rangeScale = (2.0f * M_PI); const float maxWidth = radius / 10.0f; const float minWidth = maxWidth * 0.3f; QPolygonF p(6); p.replace(0, QPointF(xRef-maxWidth/2.0f, yRef-radius * 0.5f)); p.replace(1, QPointF(xRef-minWidth/2.0f, yRef-radius * 0.9f)); p.replace(2, QPointF(xRef+minWidth/2.0f, yRef-radius * 0.9f)); p.replace(3, QPointF(xRef+maxWidth/2.0f, yRef-radius * 0.5f)); p.replace(4, QPointF(xRef, yRef-radius * 0.46f)); p.replace(5, QPointF(xRef-maxWidth/2.0f, yRef-radius * 0.5f)); rotatePolygonClockWiseRad(p, yaw*rangeScale, QPointF(xRef, yRef)); QBrush indexBrush; indexBrush.setColor(defaultColor); indexBrush.setStyle(Qt::SolidPattern); painter->setPen(Qt::SolidLine); painter->setPen(defaultColor); painter->setBrush(indexBrush); drawPolygon(p, painter); }