/** ****************************************************************************** * * @file lineardialgadgetwidget.cpp * @author Edouard Lafargue Copyright (C) 2010. * @addtogroup GCSPlugins GCS Plugins * @{ * @addtogroup LinearDialPlugin Linear Dial Plugin * @{ * @brief Implements a gadget that displays linear gauges and generic indicators *****************************************************************************/ /* * 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 "lineardialgadgetwidget.h" #include #include #include LineardialGadgetWidget::LineardialGadgetWidget(QWidget *parent) : QGraphicsView(parent) { setMinimumSize(32,32); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); setScene(new QGraphicsScene(this)); setRenderHints(QPainter::Antialiasing); m_renderer = new QSvgRenderer(); verticalDial = false; paint(); obj1 = NULL; fieldName = NULL; fieldValue = NULL; indexTarget = 0; indexValue = 0; places = 0; factor = 1; // This timer mechanism makes the index rotate smoothly connect(&dialTimer, SIGNAL(timeout()), this, SLOT(moveIndex())); dialTimer.start(30); } LineardialGadgetWidget::~LineardialGadgetWidget() { // Do nothing } /*! \brief Connects the widget to the relevant UAVObjects */ void LineardialGadgetWidget::connectInput(QString object1, QString nfield1) { if (obj1 != NULL) disconnect(obj1,SIGNAL(objectUpdated(UAVObject*)),this,SLOT(updateIndex(UAVObject*))); ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager *objManager = pm->getObject(); // qDebug() << "Lineardial Connect needles - " << object1 << "-"<< nfield1; // Check validity of arguments first, reject empty args and unknown fields. if (!(object1.isEmpty() || nfield1.isEmpty())) { obj1 = dynamic_cast( objManager->getObject(object1) ); if (obj1 != NULL ) { connect(obj1, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(updateIndex(UAVObject*))); if(nfield1.contains("-")) { QStringList fieldSubfield = nfield1.split("-", QString::SkipEmptyParts); field1 = fieldSubfield.at(0); subfield1 = fieldSubfield.at(1); haveSubField1 = true; } else { field1= nfield1; haveSubField1 = false; } if (fieldName) fieldName->setPlainText(nfield1); updateIndex(obj1); } else { qDebug() << "Error: Object is unknown (" << object1 << ") this should not happen."; } } } /*! \brief Called by the UAVObject which got updated Updates the numeric value and/or the icon if the dial wants this. */ void LineardialGadgetWidget::updateIndex(UAVObject *object1) { // Double check that the field exists: UAVObjectField* field = object1->getField(field1); if (field) { QString s; if (field->isNumeric()) { double v; if(haveSubField1){ int indexOfSubField = field->getElementNames().indexOf(QRegExp(subfield1, Qt::CaseSensitive, QRegExp::FixedString)); v = field->getDouble(indexOfSubField)*factor; }else v = field->getDouble()*factor; setIndex(v); s.sprintf("%.*f",places,v); } if (field->isText()) { s = field->getValue().toString(); if (fieldSymbol) { // If we defined a symbol, we will look for a matching // SVG element to display: if (m_renderer->elementExists("symbol-" + s)) { fieldSymbol->setElementId("symbol-" + s); } else { fieldSymbol->setElementId("symbol"); } } } if (fieldValue) fieldValue->setPlainText(s); if (index && !dialTimer.isActive()) dialTimer.start(); } else { qDebug() << "Wrong field, maybe an issue with object disconnection ?"; } } /*! \brief Setup dial using its master SVG template. Should only be called after the min/max ranges have been set. */ void LineardialGadgetWidget::setDialFile(QString dfn) { QGraphicsScene *l_scene = scene(); setBackgroundBrush(QBrush(Utils::StyleHelper::baseColor())); if (QFile::exists(dfn) && m_renderer->load(dfn) && m_renderer->isValid() ) { l_scene->clear(); // Beware: clear also deletes all objects // which are currently in the scene background = new QGraphicsSvgItem(); background->setSharedRenderer(m_renderer); background->setElementId("background"); background->setFlags(QGraphicsItem::ItemClipsChildrenToShape| QGraphicsItem::ItemClipsToShape); l_scene->addItem(background); // The red/yellow/green zones are optional, we just // test on the presence of "red" if (m_renderer->elementExists("red")) { // Order is important: red, then yellow then green // overlayed on top of each other red = new QGraphicsSvgItem(); red->setSharedRenderer(m_renderer); red->setElementId("red"); red->setParentItem(background); yellow = new QGraphicsSvgItem(); yellow->setSharedRenderer(m_renderer); yellow->setElementId("yellow"); yellow->setParentItem(background); green = new QGraphicsSvgItem(); green->setSharedRenderer(m_renderer); green->setElementId("green"); green->setParentItem(background); // In order to properly render the Green/Yellow/Red graphs, we need to find out // the starting location of the bargraph rendering area: QMatrix textMatrix = m_renderer->matrixForElement("bargraph"); qreal bgX = textMatrix.mapRect(m_renderer->boundsOnElement("bargraph")).x(); qreal bgY = textMatrix.mapRect(m_renderer->boundsOnElement("bargraph")).y(); bargraphSize = textMatrix.mapRect(m_renderer->boundsOnElement("bargraph")).width(); // Detect if the bargraph is vertical or horizontal. qreal bargraphHeight = textMatrix.mapRect(m_renderer->boundsOnElement("bargraph")).height(); if (bargraphHeight > bargraphSize) { verticalDial = true; bargraphSize = bargraphHeight; } else { verticalDial = false; } // Now adjust the red/yellow/green zones: double range = maxValue-minValue; green->resetTransform(); double greenScale = (greenMax-greenMin)/range; double greenStart = verticalDial ? (maxValue-greenMax)/range*green->boundingRect().height() : (greenMin-minValue)/range*green->boundingRect().width(); QTransform matrix; matrix.reset(); if (verticalDial) { matrix.scale(1,greenScale); matrix.translate(bgX,(greenStart+bgY)/greenScale); } else { matrix.scale(greenScale,1); matrix.translate((greenStart+bgX)/greenScale,bgY); } green->setTransform(matrix,false); yellow->resetTransform(); double yellowScale = (yellowMax-yellowMin)/range; double yellowStart = verticalDial ? (maxValue-yellowMax)/range*yellow->boundingRect().height() : (yellowMin-minValue)/range*yellow->boundingRect().width(); matrix.reset(); if (verticalDial) { matrix.scale(1,yellowScale); matrix.translate(bgX,(yellowStart+bgY)/yellowScale); } else { matrix.scale(yellowScale,1); matrix.translate((yellowStart+bgX)/yellowScale,bgY); } yellow->setTransform(matrix,false); red->resetTransform(); double redScale = (redMax-redMin)/range; double redStart = verticalDial ? (maxValue-redMax)/range*red->boundingRect().height() : (redMin-minValue)/range*red->boundingRect().width(); matrix.reset(); if (verticalDial) { matrix.scale(1,redScale); matrix.translate(bgX,(redStart+bgY)/redScale); } else { matrix.scale(redScale,1); matrix.translate((redStart+bgX)/redScale,bgY); } red->setTransform(matrix,false); } else { red = NULL; yellow = NULL; green = NULL; } // Check whether the dial wants to display a moving index: if (m_renderer->elementExists("needle")) { QMatrix textMatrix = m_renderer->matrixForElement("needle"); QRectF nRect = textMatrix.mapRect(m_renderer->boundsOnElement("needle")); startX = nRect.x(); startY = nRect.y(); QTransform matrix; matrix.translate(startX,startY); index = new QGraphicsSvgItem(); index->setSharedRenderer(m_renderer); index->setElementId("needle"); index->setTransform(matrix,false); index->setParentItem(background); } else { index = NULL; } // Check whether the dial wants display its field name: if (m_renderer->elementExists("field")) { QMatrix textMatrix = m_renderer->matrixForElement("field"); QRectF rect = textMatrix.mapRect(m_renderer->boundsOnElement("field")); qreal startX = rect.x(); qreal startY = rect.y(); qreal elHeight = rect.height(); QTransform matrix; matrix.translate(startX,startY-elHeight/2); fieldName = new QGraphicsTextItem("field"); fieldName->setFont(QFont("Arial",(int)elHeight)); fieldName->setDefaultTextColor(QColor("White")); fieldName->setTransform(matrix,false); fieldName->setParentItem(background); } else { fieldName = NULL; } // Check whether the dial wants display the numeric value: if (m_renderer->elementExists("value")) { QMatrix textMatrix = m_renderer->matrixForElement("value"); QRectF nRect = textMatrix.mapRect(m_renderer->boundsOnElement("value")); qreal startX = nRect.x(); qreal startY = nRect.y(); qreal elHeight = nRect.height(); QTransform matrix; matrix.translate(startX,startY-elHeight/2); fieldValue = new QGraphicsTextItem("0.00"); fieldValue->setFont(QFont("Arial",(int)elHeight)); fieldValue->setDefaultTextColor(QColor("White")); fieldValue->setTransform(matrix,false); fieldValue->setParentItem(background); } else { fieldValue = NULL; } // Check whether the dial wants to display the value as a // symbol (only works for text values): if (m_renderer->elementExists("symbol")) { QMatrix textMatrix = m_renderer->matrixForElement("symbol"); qreal startX = textMatrix.mapRect(m_renderer->boundsOnElement("symbol")).x(); qreal startY = textMatrix.mapRect(m_renderer->boundsOnElement("symbol")).y(); QTransform matrix; matrix.translate(startX,startY); fieldSymbol = new QGraphicsSvgItem(); fieldSymbol->setElementId("symbol"); fieldSymbol->setSharedRenderer(m_renderer); fieldSymbol->setTransform(matrix,false); fieldSymbol->setParentItem(background); } else { fieldSymbol = NULL; } if (m_renderer->elementExists("foreground")) { foreground = new QGraphicsSvgItem(); foreground->setSharedRenderer(m_renderer); foreground->setElementId("foreground"); foreground->setParentItem(background); fgenabled = true; } else { fgenabled = false; } l_scene->setSceneRect(background->boundingRect()); // Reset the current index value: indexValue = 0; if (!dialTimer.isActive() && index) dialTimer.start(); } else { qDebug() << "no file "; m_renderer->load(QString(":/lineardial/images/empty.svg")); l_scene->clear(); // This also deletes all items contained in the scene. background = new QGraphicsSvgItem(); background->setSharedRenderer(m_renderer); l_scene->addItem(background); fieldName = NULL; fieldValue = NULL; fieldSymbol = NULL; index = NULL; } } void LineardialGadgetWidget::setDialFont(QString fontProps) { // Note: a bit of juggling to preserve the automatic // font size which was calculated upon dial initialization. QFont font = QFont("Arial",12); font.fromString(fontProps); if (fieldName) { int fieldSize = fieldName->font().pointSize(); font.setPointSize(fieldSize); fieldName->setFont(font); } if (fieldValue) { int fieldSize = fieldValue->font().pointSize(); font.setPointSize(fieldSize); fieldValue->setFont(font); } } void LineardialGadgetWidget::paint() { update(); } void LineardialGadgetWidget::paintEvent(QPaintEvent *event) { // Skip painting until the dial file is loaded if (! m_renderer->isValid()) { qDebug()<<"Dial file not loaded, not rendering"; return; } QGraphicsView::paintEvent(event); } // This event enables the dial to be dynamically resized // whenever the gadget is resized, taking advantage of the vector // nature of SVG dials. void LineardialGadgetWidget::resizeEvent(QResizeEvent *event) { fitInView(background, Qt::KeepAspectRatio ); } // Converts the value into an percentage: // this enables smooth movement in moveIndex below void LineardialGadgetWidget::setIndex(double value) { if (verticalDial) { indexTarget = 100*(maxValue-value)/(maxValue-minValue); } else { indexTarget = 100*(value-minValue)/(maxValue-minValue); } } // Take an input value and move the index accordingly // Move is smooth, starts fast and slows down when // approaching the target. void LineardialGadgetWidget::moveIndex() { if (!index) { // Safeguard dialTimer.stop(); return; } if ((abs((indexValue-indexTarget)*10) > 3)) { indexValue += (indexTarget - indexValue)/5; } else { indexValue = indexTarget; dialTimer.stop(); } QTransform matrix; index->resetTransform(); qreal trans = indexValue*bargraphSize/100; if (verticalDial) { matrix.translate(startX,trans+startY); } else { matrix.translate(trans+startX,startY); } index->setTransform(matrix,false); update(); }