/** ****************************************************************************** * * @file scopegadgetwidget.cpp * @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010. * @addtogroup GCSPlugins GCS Plugins * @{ * @addtogroup ScopePlugin Scope Gadget Plugin * @{ * @brief The scope Gadget, graphically plots the states of UAVObjects *****************************************************************************/ /* * 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 "scopegadgetwidget.h" #include "utils/stylehelper.h" #include "extensionsystem/pluginmanager.h" #include "uavobjectmanager.h" #include "uavobject.h" #include "coreplugin/icore.h" #include "coreplugin/connectionmanager.h" #include "qwt/src/qwt_plot_curve.h" #include "qwt/src/qwt_legend.h" #include "qwt/src/qwt_legend_item.h" #include "qwt/src/qwt_plot_grid.h" #include #include #include #include #include #include #include #include #include #include #include ScopeGadgetWidget::ScopeGadgetWidget(QWidget *parent) : QwtPlot(parent) { setMouseTracking(true); // canvas()->setMouseTracking(true); // Setup the timer that replots data replotTimer = new QTimer(this); connect(replotTimer, SIGNAL(timeout()), this, SLOT(replotNewData())); // Listen to telemetry connection/disconnection events, no point in // running the scopes if we are not connected and not replaying logs. // Also listen to disconnect actions from the user Core::ConnectionManager *cm = Core::ICore::instance()->connectionManager(); connect(cm, SIGNAL(deviceAboutToDisconnect()), this, SLOT(stopPlotting())); connect(cm, SIGNAL(deviceConnected(QIODevice *)), this, SLOT(startPlotting())); m_csvLoggingStarted = 0; m_csvLoggingEnabled = 0; m_csvLoggingHeaderSaved = 0; m_csvLoggingDataSaved = 0; m_csvLoggingDataUpdated = 0; m_csvLoggingNameSet = 0; m_csvLoggingConnected = 0; m_csvLoggingNewFileOnConnect = 0; m_csvLoggingPath = QString("./csvlogging/"); m_csvLoggingStartTime = QDateTime::currentDateTime(); // Listen to autopilot connection events connect(cm, SIGNAL(deviceAboutToDisconnect()), this, SLOT(csvLoggingDisconnect())); connect(cm, SIGNAL(deviceConnected(QIODevice *)), this, SLOT(csvLoggingConnect())); } ScopeGadgetWidget::~ScopeGadgetWidget() { if (replotTimer) { replotTimer->stop(); delete replotTimer; replotTimer = NULL; } // Get the object to de-monitor ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager *objManager = pm->getObject(); foreach(QString uavObjName, m_connectedUAVObjects) { UAVDataObject *obj = dynamic_cast(objManager->getObject(uavObjName)); disconnect(obj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(uavObjectReceived(UAVObject *))); } clearCurvePlots(); } void ScopeGadgetWidget::mousePressEvent(QMouseEvent *e) { QwtPlot::mousePressEvent(e); } void ScopeGadgetWidget::mouseReleaseEvent(QMouseEvent *e) { QwtPlot::mouseReleaseEvent(e); } void ScopeGadgetWidget::mouseDoubleClickEvent(QMouseEvent *e) { // On double-click, toggle legend mutex.lock(); if (legend()) { deleteLegend(); } else { addLegend(); } mutex.unlock(); // On double-click, reset plot zoom setAxisAutoScale(QwtPlot::yLeft, true); update(); QwtPlot::mouseDoubleClickEvent(e); } void ScopeGadgetWidget::mouseMoveEvent(QMouseEvent *e) { QwtPlot::mouseMoveEvent(e); } void ScopeGadgetWidget::wheelEvent(QWheelEvent *e) { // Change zoom on scroll wheel event QwtInterval yInterval = axisInterval(QwtPlot::yLeft); if (yInterval.minValue() != yInterval.maxValue()) { // Make sure that the two values are never the same. Sometimes axisInterval returns (0,0) // Determine what y value to zoom about. NOTE, this approach has a bug that the in that // the value returned by Qt includes the legend, whereas the value transformed by Qwt // does *not*. Thus, when zooming with a legend, there will always be a small bias error. // In practice, this seems not to be a UI problem. QPoint mouse_pos = e->pos(); // Get the mouse coordinate in the frame double zoomLine = invTransform(QwtPlot::yLeft, mouse_pos.y()); // Transform the y mouse coordinate into a frame value. double zoomScale = 1.1; // THIS IS AN ARBITRARY CONSTANT, AND PERHAPS SHOULD BE IN A DEFINE INSTEAD OF BURIED HERE mutex.lock(); // DOES THIS mutex.lock NEED TO BE HERE? I DON'T KNOW, I JUST COPIED IT FROM THE ABOVE CODE // Set the scale if (e->delta() < 0) { setAxisScale(QwtPlot::yLeft, (yInterval.minValue() - zoomLine) * zoomScale + zoomLine, (yInterval.maxValue() - zoomLine) * zoomScale + zoomLine); } else { setAxisScale(QwtPlot::yLeft, (yInterval.minValue() - zoomLine) / zoomScale + zoomLine, (yInterval.maxValue() - zoomLine) / zoomScale + zoomLine); } mutex.unlock(); } QwtPlot::wheelEvent(e); } void ScopeGadgetWidget::showEvent(QShowEvent *e) { replotNewData(); QwtPlot::showEvent(e); } /** * Starts/stops telemetry */ void ScopeGadgetWidget::startPlotting() { if (replotTimer && !replotTimer->isActive()) { replotTimer->start(m_refreshInterval); } } void ScopeGadgetWidget::stopPlotting() { if (replotTimer) { replotTimer->stop(); } } void ScopeGadgetWidget::deleteLegend() { if (legend()) { disconnect(this, SIGNAL(legendChecked(QwtPlotItem *, bool)), this, 0); insertLegend(NULL, QwtPlot::TopLegend); } } void ScopeGadgetWidget::addLegend() { if (legend()) { return; } // Show a legend at the top QwtLegend *legend = new QwtLegend(); legend->setItemMode(QwtLegend::CheckableItem); legend->setFrameStyle(QFrame::Box | QFrame::Sunken); legend->setToolTip(tr("Click legend to show/hide scope trace.\nDouble click legend or plot to show/hide legend.")); // set colors QPalette pal = legend->palette(); pal.setColor(legend->backgroundRole(), QColor(100, 100, 100)); pal.setColor(QPalette::Text, QColor(0, 0, 0)); legend->setPalette(pal); insertLegend(legend, QwtPlot::TopLegend); // Update the checked/unchecked state of the legend items // -> this is necessary when hiding a legend where some plots are // not visible, and then un-hiding it. foreach(QwtPlotItem * item, this->itemList()) { bool on = item->isVisible(); QWidget *w = legend->find(item); if (w && w->inherits("QwtLegendItem")) { ((QwtLegendItem *)w)->setChecked(!on); } } connect(this, SIGNAL(legendChecked(QwtPlotItem *, bool)), this, SLOT(showCurve(QwtPlotItem *, bool))); } void ScopeGadgetWidget::preparePlot(PlotType plotType) { m_plotType = plotType; clearCurvePlots(); setMinimumSize(64, 64); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); setCanvasBackground(QColor(64, 64, 64)); // Add grid lines QwtPlotGrid *grid = new QwtPlotGrid; grid->setMajPen(QPen(Qt::gray, 0, Qt::DashLine)); grid->setMinPen(QPen(Qt::lightGray, 0, Qt::DotLine)); grid->setPen(QPen(Qt::darkGray, 1, Qt::DotLine)); grid->attach(this); // Only start the timer if we are already connected Core::ConnectionManager *cm = Core::ICore::instance()->connectionManager(); if (cm->isConnected() && replotTimer) { if (!replotTimer->isActive()) { replotTimer->start(m_refreshInterval); } else { replotTimer->setInterval(m_refreshInterval); } } } void ScopeGadgetWidget::showCurve(QwtPlotItem *item, bool on) { item->setVisible(!on); if (legend()) { QWidget *w = legend()->find(item); if (w && w->inherits("QwtLegendItem")) { ((QwtLegendItem *)w)->setChecked(on); } } mutex.lock(); replot(); mutex.unlock(); } void ScopeGadgetWidget::setupSequentialPlot() { preparePlot(SequentialPlot); setAxisScaleDraw(QwtPlot::xBottom, new QwtScaleDraw()); setAxisScale(QwtPlot::xBottom, 0, m_plotDataSize); setAxisLabelRotation(QwtPlot::xBottom, 0.0); setAxisLabelAlignment(QwtPlot::xBottom, Qt::AlignLeft | Qt::AlignBottom); QwtScaleWidget *scaleWidget = axisWidget(QwtPlot::xBottom); // reduce the gap between the scope canvas and the axis scale scaleWidget->setMargin(0); // reduce the axis font size QFont fnt(axisFont(QwtPlot::xBottom)); fnt.setPointSize(7); setAxisFont(QwtPlot::xBottom, fnt); // x-axis setAxisFont(QwtPlot::yLeft, fnt); // y-axis } void ScopeGadgetWidget::setupChronoPlot() { preparePlot(ChronoPlot); setAxisScaleDraw(QwtPlot::xBottom, new TimeScaleDraw()); uint NOW = QDateTime::currentDateTime().toTime_t(); setAxisScale(QwtPlot::xBottom, NOW - m_plotDataSize / 1000, NOW); setAxisLabelRotation(QwtPlot::xBottom, 0.0); setAxisLabelAlignment(QwtPlot::xBottom, Qt::AlignLeft | Qt::AlignBottom); QwtScaleWidget *scaleWidget = axisWidget(QwtPlot::xBottom); // reduce the gap between the scope canvas and the axis scale scaleWidget->setMargin(0); // reduce the axis font size QFont fnt(axisFont(QwtPlot::xBottom)); fnt.setPointSize(7); setAxisFont(QwtPlot::xBottom, fnt); // x-axis setAxisFont(QwtPlot::yLeft, fnt); // y-axis } void ScopeGadgetWidget::addCurvePlot(QString objectName, QString fieldPlusSubField, int scaleFactor, int meanSamples, QString mathFunction, QPen pen, bool antialiased) { QString fieldName = fieldPlusSubField; QString elementName; int element = 0; if (fieldPlusSubField.contains("-")) { QStringList fieldSubfield = fieldName.split("-", QString::SkipEmptyParts); fieldName = fieldSubfield.at(0); elementName = fieldSubfield.at(1); } // Create the curve QString curveName = objectName + "." + fieldName; if (!elementName.isEmpty()) { curveName.append("." + elementName); } // Get the uav object ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager *objManager = pm->getObject(); UAVDataObject *object = dynamic_cast(objManager->getObject(objectName)); if (!object) { qDebug() << "Object" << objectName << "is missing"; return; } UAVObjectField *field = object->getField(fieldName); if (!field) { qDebug() << "In scope gadget, in fields loaded from GCS config file, field" << fieldName << "of object" << objectName << "is missing"; return; } if (!elementName.isEmpty()) { element = field->getElementNames().indexOf(QRegExp(elementName, Qt::CaseSensitive, QRegExp::FixedString)); if (element < 0) { qDebug() << "In scope gadget, in fields loaded from GCS config file, field" << fieldName << "of object" << objectName << "element name" << elementName << "is missing"; return; } } QString units = field->getUnits(); if (units == 0) { units = QString(); } QString curveNameScaled; if (scaleFactor == 0) { curveNameScaled = curveName + " (" + units + ")"; } else { curveNameScaled = curveName + " (x10^" + QString::number(scaleFactor) + " " + units + ")"; } QwtPlotCurve *plotCurve = new QwtPlotCurve(curveNameScaled); if (antialiased) { plotCurve->setRenderHint(QwtPlotCurve::RenderAntialiased); } plotCurve->setPen(pen); plotCurve->attach(this); PlotData *plotData; if (m_plotType == SequentialPlot) { plotData = new SequentialPlotData(object, field, element, plotCurve, scaleFactor, meanSamples, mathFunction, m_plotDataSize); } else if (m_plotType == ChronoPlot) { plotData = new ChronoPlotData(object, field, element, plotCurve, scaleFactor, meanSamples, mathFunction, m_plotDataSize); } // If the y-bounds are supplied, set them if (plotData->yMin() != plotData->yMax()) { setAxisScale(QwtPlot::yLeft, plotData->yMin(), plotData->yMax()); } // Keep the curve details for later m_curvesData.insert(curveNameScaled, plotData); // Link to the new signal data only if this UAVObject has not been connected yet if (!m_connectedUAVObjects.contains(object->getName())) { m_connectedUAVObjects.append(object->getName()); connect(object, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(uavObjectReceived(UAVObject *))); } mutex.lock(); replot(); mutex.unlock(); } void ScopeGadgetWidget::uavObjectReceived(UAVObject *obj) { foreach(PlotData * plotData, m_curvesData.values()) { if (plotData->append(obj)) { m_csvLoggingDataUpdated = 1; } } csvLoggingAddData(); } void ScopeGadgetWidget::replotNewData() { if (!isVisible()) { return; } QMutexLocker locker(&mutex); foreach(PlotData * plotData, m_curvesData.values()) { plotData->removeStaleData(); plotData->updatePlotCurveData(); } QDateTime NOW = QDateTime::currentDateTime(); double toTime = NOW.toTime_t(); toTime += NOW.time().msec() / 1000.0; if (m_plotType == ChronoPlot) { setAxisScale(QwtPlot::xBottom, toTime - m_plotDataSize, toTime); } csvLoggingInsertData(); replot(); } void ScopeGadgetWidget::clearCurvePlots() { foreach(PlotData * plotData, m_curvesData.values()) { delete plotData; } m_curvesData.clear(); } void ScopeGadgetWidget::saveState(QSettings *qSettings) { /* // plot state int i = 1; foreach(PlotData * plotData, m_curvesData.values()) { bool plotVisible = plotData->isVisible(); if (!plotVisible) { qSettings->setValue(QString("plot%1").arg(i), plotVisible); } i++; } // legend state qSettings->setValue("legendVisible", legend() != NULL); */ } void ScopeGadgetWidget::restoreState(QSettings *qSettings) { /* // plot state int i = 1; foreach(PlotData * plotData, m_curvesData.values()) { bool visible = qSettings->value(QString("plot%1").arg(i), true).toBool(); showCurve(plotData->m_plotCurve, !visible); i++; } // legend state bool legendVisible = qSettings->value("legendVisible", true).toBool(); if (legendVisible) { addLegend(); } else { deleteLegend(); } */ } /* int csvLoggingEnable; int csvLoggingHeaderSaved; int csvLoggingDataSaved; QString csvLoggingPath; QFile csvLoggingFile; */ int ScopeGadgetWidget::csvLoggingStart() { if (!m_csvLoggingStarted) { if (m_csvLoggingEnabled) { if ((!m_csvLoggingNewFileOnConnect) || (m_csvLoggingNewFileOnConnect && m_csvLoggingConnected)) { QDateTime NOW = QDateTime::currentDateTime(); m_csvLoggingStartTime = NOW; m_csvLoggingHeaderSaved = 0; m_csvLoggingDataSaved = 0; m_csvLoggingBuffer.clear(); QDir PathCheck(m_csvLoggingPath); if (!PathCheck.exists()) { PathCheck.mkpath("./"); } if (m_csvLoggingNameSet) { m_csvLoggingFile.setFileName(QString("%1/%2_%3_%4.csv").arg(m_csvLoggingPath).arg(m_csvLoggingName).arg(NOW.toString("yyyy-MM-dd")).arg(NOW.toString("hh-mm-ss"))); } else { m_csvLoggingFile.setFileName(QString("%1/Log_%2_%3.csv").arg(m_csvLoggingPath).arg(NOW.toString("yyyy-MM-dd")).arg(NOW.toString("hh-mm-ss"))); } QDir FileCheck(m_csvLoggingFile.fileName()); if (FileCheck.exists()) { m_csvLoggingFile.setFileName(""); } else { m_csvLoggingStarted = 1; csvLoggingInsertHeader(); } } } } return 0; } int ScopeGadgetWidget::csvLoggingStop() { m_csvLoggingStarted = 0; return 0; } int ScopeGadgetWidget::csvLoggingInsertHeader() { if (!m_csvLoggingStarted) { return -1; } if (m_csvLoggingHeaderSaved) { return -2; } if (m_csvLoggingDataSaved) { return -3; } m_csvLoggingHeaderSaved = 1; if (m_csvLoggingFile.open(QIODevice::WriteOnly | QIODevice::Append) == false) { qDebug() << "Unable to open " << m_csvLoggingFile.fileName() << " for csv logging Header"; } else { QTextStream ts(&m_csvLoggingFile); ts << "date" << ", " << "Time" << ", " << "Sec since start" << ", " << "Connected" << ", " << "Data changed"; foreach(PlotData * plotData2, m_curvesData.values()) { ts << ", "; ts << plotData2->objectName(); ts << "." << plotData2->field()->getName(); if (!plotData2->elementName().isEmpty()) { ts << "." << plotData2->elementName(); } } ts << endl; m_csvLoggingFile.close(); } return 0; } int ScopeGadgetWidget::csvLoggingAddData() { if (!m_csvLoggingStarted) { return -1; } m_csvLoggingDataValid = false; QDateTime NOW = QDateTime::currentDateTime(); QString tempString; QTextStream ss(&tempString); ss << NOW.toString("yyyy-MM-dd") << ", " << NOW.toString("hh:mm:ss.z") << ", "; #if QT_VERSION >= 0x040700 ss << (NOW.toMSecsSinceEpoch() - m_csvLoggingStartTime.toMSecsSinceEpoch()) / 1000.00; #else ss << (NOW.toTime_t() - m_csvLoggingStartTime.toTime_t()); #endif ss << ", " << m_csvLoggingConnected << ", " << m_csvLoggingDataUpdated; m_csvLoggingDataUpdated = false; foreach(PlotData * plotData2, m_curvesData.values()) { ss << ", "; if (plotData2->hasData()) { ss << QString().sprintf("%3.10g", plotData2->lastData()); m_csvLoggingDataValid = true; } } ss << endl; if (m_csvLoggingDataValid) { QTextStream ts(&m_csvLoggingBuffer); ts << tempString; } return 0; } int ScopeGadgetWidget::csvLoggingInsertData() { if (!m_csvLoggingStarted) { return -1; } m_csvLoggingDataSaved = 1; if (m_csvLoggingFile.open(QIODevice::WriteOnly | QIODevice::Append) == false) { qDebug() << "Unable to open " << m_csvLoggingFile.fileName() << " for csv logging Data"; } else { QTextStream ts(&m_csvLoggingFile); ts << m_csvLoggingBuffer; m_csvLoggingFile.close(); } m_csvLoggingBuffer.clear(); return 0; } void ScopeGadgetWidget::csvLoggingSetName(QString newName) { m_csvLoggingName = newName; m_csvLoggingNameSet = 1; } void ScopeGadgetWidget::csvLoggingConnect() { m_csvLoggingConnected = 1; if (m_csvLoggingNewFileOnConnect) { csvLoggingStart(); } } void ScopeGadgetWidget::csvLoggingDisconnect() { m_csvLoggingHeaderSaved = 0; m_csvLoggingConnected = 0; if (m_csvLoggingNewFileOnConnect) { csvLoggingStop(); } }