/** ****************************************************************************** * * @file pfdgadgetwidget.cpp * @author Edouard Lafargue Copyright (C) 2010. * @brief * @see The GNU Public License (GPL) Version 3 * @defgroup pfd * @{ * *****************************************************************************/ /* * This program 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. * * This program 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 this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "pfdgadgetwidget.h" #include #include PFDGadgetWidget::PFDGadgetWidget(QWidget *parent) : QGraphicsView(parent) { // TODO: create a proper "needle" object instead of hardcoding all this // which is ugly (but easy). setMinimumSize(64,64); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); setScene(new QGraphicsScene(this)); setRenderHints(QPainter::Antialiasing); m_renderer = new QSvgRenderer(); attitudeObj = NULL; headingObj = NULL; compassBandWidth = 0; /* obj2 = NULL; obj3 = NULL; */ // This timer mechanism makes needles rotate smoothly connect(&dialTimer, SIGNAL(timeout()), this, SLOT(rotateNeedles())); dialTimer.start(20); } PFDGadgetWidget::~PFDGadgetWidget() { // Do nothing } /*! \brief Connects the widget to the relevant UAVObjects We want: AttitudeActual, FlightBattery, Location */ void PFDGadgetWidget::connectNeedles() { if (attitudeObj != NULL) disconnect(attitudeObj,SIGNAL(objectUpdated(UAVObject*)),this,SLOT(updateAttitude(UAVObject*))); if (headingObj != NULL) disconnect(headingObj,SIGNAL(objectUpdated(UAVObject*)),this,SLOT(updateHeading(UAVObject*))); ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager *objManager = pm->getObject(); attitudeObj = dynamic_cast(objManager->getObject("AttitudeActual")); if (attitudeObj != NULL ) { connect(attitudeObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(updateAttitude(UAVObject*))); } else { std::cout << "Error: Object is unknown (AttitudeActual)." << std::endl; } headingObj = dynamic_cast(objManager->getObject("PositionActual")); if (headingObj != NULL ) { connect(headingObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(updateHeading(UAVObject*))); } else { std::cout << "Error: Object is unknown (PositionActual)." << std::endl; } } /*! \brief Reads the updated attitude and computes the new display position */ void PFDGadgetWidget::updateAttitude(UAVObject *object1) { // Double check that the field exists: QString roll = QString("Roll"); QString pitch = QString("Pitch"); UAVObjectField* field = object1->getField(roll); UAVObjectField* field2 = object1->getField(pitch); if (field && field2) { // These factors assume some things about the PFD SVG, namely: // - Roll value in degrees // - Pitch lines are 300px high for a +20/-20 range, which means // 7.5 pixels per pitch degree. rollTarget = field->getDouble()*(-1); pitchTarget = field2->getDouble()*7.5; if (!dialTimer.isActive()) dialTimer.start(); // Rearm the dial Timer which might be stopped. } else { std::cout << "UpdateAttitude: Wrong field, maybe an issue with object disconnection ?" << std::endl; } } /*! \brief Called by the UAVObject which got updated */ void PFDGadgetWidget::updateHeading(UAVObject *object1) { // Double check that the field exists: QString heading = QString("Heading"); UAVObjectField* field = object1->getField(heading); if (field) { // These factors assume some things about the PFD SVG, namely: // - Heading value in degrees // - Scale is 540 degrees large headingTarget = field->getDouble()*compassBandWidth/(-540); if (!dialTimer.isActive()) dialTimer.start(); // Rearm the dial Timer which might be stopped. } else { std::cout << "UpdateHeading: Wrong field, maybe an issue with object disconnection ?" << std::endl; } } /*! \brief Called by the UAVObject which got updated */ void PFDGadgetWidget::updateAirspeed(UAVObject *object3) { } /*! \brief Called by the UAVObject which got updated */ void PFDGadgetWidget::updateAltitude(UAVObject *object3) { } /*! \brief Called by the UAVObject which got updated */ void PFDGadgetWidget::updateBattery(UAVObject *object3) { } /* Initializes the dial file, and does all the one-time calculations for display later. */ void PFDGadgetWidget::setDialFile(QString dfn) { if (QFile::exists(dfn)) { m_renderer->load(dfn); if(m_renderer->isValid()) { /* The PFD element IDs are fixed, not like with the analog dial. - Background: background - Foreground: foreground (contains all fixed elements, including plane) - earth/sky : world - Roll scale: rollscale - compass frame: compass (part of the foreground) - compass band : compass-band - Home point: homewaypoint - Next point: nextwaypoint - Home point bearing: homewaypoint-bearing - Next point bearing: nextwaypoint-bearing */ QGraphicsScene *l_scene = scene(); l_scene->clear(); // Deletes all items contained in the scene as well. m_background = new QGraphicsSvgItem(); // All other items will be clipped to the shape of the background m_background->setFlags(QGraphicsItem::ItemClipsChildrenToShape| QGraphicsItem::ItemClipsToShape); m_background->setSharedRenderer(m_renderer); m_background->setElementId("background"); l_scene->addItem(m_background); m_world = new QGraphicsSvgItem(); m_world->setParentItem(m_background); m_world->setSharedRenderer(m_renderer); m_world->setElementId("world"); l_scene->addItem(m_world); // red Roll scale: rollscale m_rollscale = new QGraphicsSvgItem(); m_rollscale->setSharedRenderer(m_renderer); m_rollscale->setElementId("rollscale"); l_scene->addItem(m_rollscale); // Home point: m_homewaypoint = new QGraphicsSvgItem(); // Next point: m_nextwaypoint = new QGraphicsSvgItem(); // Home point bearing: m_homepointbearing = new QGraphicsSvgItem(); // Next point bearing: m_nextpointbearing = new QGraphicsSvgItem(); m_foreground = new QGraphicsSvgItem(); m_foreground->setParentItem(m_background); m_foreground->setSharedRenderer(m_renderer); m_foreground->setElementId("foreground"); l_scene->addItem(m_foreground); // Compass: // Get the default location of the Compass: QMatrix compassMatrix = m_renderer->matrixForElement("compass"); qreal startX = compassMatrix.mapRect(m_renderer->boundsOnElement("compass")).x(); qreal startY = compassMatrix.mapRect(m_renderer->boundsOnElement("compass")).y(); // Then once we have the initial location, we can put it // into a QGraphicsSvgItem which we will display at the same // place: we do this so that the heading scale can be clipped to // the compass dial region. m_compass = new QGraphicsSvgItem(); m_compass->setSharedRenderer(m_renderer); m_compass->setElementId("compass"); m_compass->setFlags(QGraphicsItem::ItemClipsChildrenToShape| QGraphicsItem::ItemClipsToShape); l_scene->addItem(m_compass); QTransform matrix; matrix.translate(startX,startY); m_compass->setTransform(matrix,false); // Now place the compass scale inside: m_compassband = new QGraphicsSvgItem(); m_compassband->setSharedRenderer(m_renderer); m_compassband->setElementId("compass-band"); m_compassband->setParentItem(m_compass); l_scene->addItem(m_compassband); matrix.reset(); // Note: the compass band has to be a path, which means all text elements have to be // converted, ortherwise boundsOnElement does not compute the height correctly // if the highest element is a text element. This is a Qt Bug as far as I can tell. // compass-scale is the while bottom line inside the band: using the band's width // includes half the width of the letters, which causes errors: compassBandWidth = m_renderer->boundsOnElement("compass-scale").width(); l_scene->setSceneRect(m_background->boundingRect()); // Now Initialize the center for all transforms of the relevant elements to the // center of the background: // 1) Move the center of the needle to the center of the background. QRectF rectB = m_background->boundingRect(); QRectF rectN = m_world->boundingRect(); m_world->setPos(rectB.width()/2-rectN.width()/2,rectB.height()/2-rectN.height()/2); // 2) Put the transform origin point of the needle at its center. m_world->setTransformOriginPoint(rectN.width()/2,rectN.height()/2); rectN = m_rollscale->boundingRect(); m_rollscale->setPos(rectB.width()/2-rectN.width()/2,rectB.height()/2-rectN.height()/2); m_rollscale->setTransformOriginPoint(rectN.width()/2,rectN.height()/2); // Also to the same init for the compass: rectB = m_compass->boundingRect(); rectN = m_compassband->boundingRect(); m_compassband->setPos(rectB.width()/2-rectN.width()/2,rectB.height()/2-rectN.height()/2); m_compassband->setTransformOriginPoint(rectN.width()/2,rectN.height()/2); // Last: we just loaded the dial file which is by default valid for a "zero" value // of the needles, so we have to reset the needles too upon dial file loading, otherwise // we would end up with an offset when we change a dial file and the needle value // is not zero at that time. rollValue = 0; pitchValue = 0; headingValue = 0; if (!dialTimer.isActive()) dialTimer.start(); // Rearm the dial Timer which might be stopped. } } else { std::cout<<"no file: "<isValid()) { std::cout<<"Dial file not loaded, not rendering"< 5)) { double rollDiff; rollDiff =(rollTarget - rollValue)/5; m_world->setRotation(m_world->rotation()+rollDiff); m_rollscale->setRotation(m_rollscale->rotation()+rollDiff); rollValue += rollDiff; } else { // This avoids any sort of offset resulting from rounding // errors on the dichotomy above rollValue = rollTarget; m_world->setRotation(rollValue); m_rollscale->setRotation(rollValue); dialCount--; } if ((abs((pitchValue-pitchTarget)*10) > 5)) { double pitchDiff; pitchDiff = (pitchTarget - pitchValue)/5; QPointF opd = QPointF(0,pitchDiff); m_world->setTransform(QTransform::fromTranslate(opd.x(),opd.y()), true); QPointF oop = m_world->transformOriginPoint(); m_world->setTransformOriginPoint((oop.x()-opd.x()),(oop.y()-opd.y())); pitchValue += pitchDiff; } else { pitchValue = pitchTarget; // TODO: should maybe reset the transformOriginpoint here one last // time with the pitchValue=pitchTarget ? To be checked later. dialCount--; } double headingOffset = 0; double headingDiff; if ((abs((headingValue-headingTarget)*10) > 5)) { headingDiff = (headingTarget - headingValue)/5; } else { headingDiff = headingTarget-headingValue; dialCount--; } double threshold = -180*compassBandWidth/540; // Note: rendering can jump oh so very slightly when crossing the 180 degree // boundary, should not impact actual useability of the display. if ((headingValue < threshold) && ((headingValue+headingDiff)>=threshold)) { // We went over 180°: activate a -360 degree offset headingOffset = 2*threshold; } else if ((headingValue >= threshold) && ((headingValue+headingDiff)setTransform(QTransform::fromTranslate(opd.x(),opd.y()), true); headingValue += headingDiff; update(); if (!dialCount) dialTimer.stop(); }