2010-08-10 07:42:39 +02:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
*
|
|
|
|
* @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
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2011-01-31 03:09:44 +01:00
|
|
|
#include <QDir>
|
2010-08-10 07:42:39 +02:00
|
|
|
#include "scopegadgetwidget.h"
|
2010-08-26 21:54:23 +02:00
|
|
|
#include "utils/stylehelper.h"
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2011-01-20 23:30:49 +01:00
|
|
|
#include "uavtalk/telemetrymanager.h"
|
|
|
|
#include "extensionsystem/pluginmanager.h"
|
2011-01-22 18:38:22 +01:00
|
|
|
#include "uavobjectmanager.h"
|
|
|
|
#include "uavobject.h"
|
2011-01-20 23:30:49 +01:00
|
|
|
#include "coreplugin/icore.h"
|
|
|
|
#include "coreplugin/connectionmanager.h"
|
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
#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 <iostream>
|
|
|
|
#include <math.h>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QColor>
|
|
|
|
#include <QStringList>
|
|
|
|
#include <QtGui/QWidget>
|
|
|
|
#include <QtGui/QVBoxLayout>
|
|
|
|
#include <QtGui/QPushButton>
|
2011-02-14 11:19:18 +01:00
|
|
|
#include <QMutexLocker>
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2011-01-20 23:30:49 +01:00
|
|
|
using namespace Core;
|
2010-08-10 07:42:39 +02:00
|
|
|
|
|
|
|
TestDataGen* ScopeGadgetWidget::testDataGen;
|
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
// ******************************************************************
|
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
ScopeGadgetWidget::ScopeGadgetWidget(QWidget *parent) : QwtPlot(parent)
|
|
|
|
{
|
2010-09-01 23:29:24 +02:00
|
|
|
//if(testDataGen == 0)
|
|
|
|
// testDataGen = new TestDataGen();
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
setMouseTracking(true);
|
|
|
|
// canvas()->setMouseTracking(true);
|
|
|
|
|
2010-08-14 22:25:55 +02:00
|
|
|
//Setup the timer that replots data
|
|
|
|
replotTimer = new QTimer(this);
|
|
|
|
connect(replotTimer, SIGNAL(timeout()), this, SLOT(replotNewData()));
|
2011-01-20 23:30:49 +01:00
|
|
|
|
|
|
|
// Listen to telemetry connection/disconnection events, no point
|
|
|
|
// 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();
|
2011-03-25 11:59:24 +01:00
|
|
|
connect(cm, SIGNAL(deviceAboutToDisconnect()), this, SLOT(stopPlotting()));
|
2011-01-26 23:21:54 +01:00
|
|
|
connect(cm, SIGNAL(deviceConnected(QIODevice*)), this, SLOT(startPlotting()));
|
2011-01-20 23:30:49 +01:00
|
|
|
|
2011-01-31 03:09:44 +01:00
|
|
|
m_csvLoggingStarted=0;
|
|
|
|
m_csvLoggingEnabled=0;
|
|
|
|
m_csvLoggingHeaderSaved=0;
|
|
|
|
m_csvLoggingDataSaved=0;
|
2011-02-01 18:43:30 +01:00
|
|
|
m_csvLoggingDataUpdated=0;
|
2011-01-31 03:09:44 +01:00
|
|
|
m_csvLoggingNameSet=0;
|
|
|
|
m_csvLoggingConnected=0;
|
|
|
|
m_csvLoggingNewFileOnConnect=0;
|
|
|
|
m_csvLoggingPath = QString("./csvlogging/");
|
|
|
|
m_csvLoggingStartTime = QDateTime::currentDateTime();
|
|
|
|
|
|
|
|
//Listen to autopilot connection events
|
2011-03-25 11:59:24 +01:00
|
|
|
connect(cm, SIGNAL(deviceAboutToDisconnect()), this, SLOT(csvLoggingDisconnect()));
|
2011-01-31 03:09:44 +01:00
|
|
|
connect(cm, SIGNAL(deviceConnected(QIODevice*)), this, SLOT(csvLoggingConnect()));
|
2011-02-14 11:19:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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<UAVObjectManager>();
|
|
|
|
foreach (QString uavObjName, m_connectedUAVObjects)
|
|
|
|
{
|
|
|
|
UAVDataObject *obj = dynamic_cast<UAVDataObject*>(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)
|
|
|
|
{
|
|
|
|
mutex.lock();
|
|
|
|
if (legend())
|
|
|
|
deleteLegend();
|
|
|
|
else
|
|
|
|
addLegend();
|
|
|
|
mutex.unlock();
|
|
|
|
update();
|
|
|
|
|
|
|
|
QwtPlot::mouseDoubleClickEvent(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::mouseMoveEvent(QMouseEvent *e)
|
|
|
|
{
|
|
|
|
QwtPlot::mouseMoveEvent(e);
|
|
|
|
}
|
2011-01-31 03:09:44 +01:00
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
void ScopeGadgetWidget::wheelEvent(QWheelEvent *e)
|
|
|
|
{
|
|
|
|
QwtPlot::wheelEvent(e);
|
2011-01-20 23:30:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Starts/stops telemetry
|
|
|
|
*/
|
2011-01-26 23:21:54 +01:00
|
|
|
void ScopeGadgetWidget::startPlotting()
|
2011-01-20 23:30:49 +01:00
|
|
|
{
|
2011-02-27 11:26:50 +01:00
|
|
|
if (!replotTimer)
|
|
|
|
return;
|
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
if (!replotTimer->isActive())
|
2011-01-20 23:30:49 +01:00
|
|
|
replotTimer->start(m_refreshInterval);
|
|
|
|
}
|
|
|
|
|
2011-01-26 23:21:54 +01:00
|
|
|
void ScopeGadgetWidget::stopPlotting()
|
2011-01-20 23:30:49 +01:00
|
|
|
{
|
2011-02-27 11:26:50 +01:00
|
|
|
if (!replotTimer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
replotTimer->stop();
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
void ScopeGadgetWidget::deleteLegend()
|
|
|
|
{
|
|
|
|
if (!legend())
|
|
|
|
return;
|
|
|
|
|
|
|
|
disconnect(this, SIGNAL(legendChecked(QwtPlotItem *, bool)), this, 0);
|
|
|
|
|
|
|
|
insertLegend(NULL, QwtPlot::TopLegend);
|
2011-02-27 11:26:50 +01:00
|
|
|
// insertLegend(NULL, QwtPlot::ExternalLegend);
|
2011-02-14 11:19:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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"));
|
|
|
|
|
|
|
|
QPalette pal = legend->palette();
|
|
|
|
pal.setColor(legend->backgroundRole(), QColor(100, 100, 100)); // background colour
|
2011-02-27 11:26:50 +01:00
|
|
|
// pal.setColor(legend->backgroundRole(), Qt::transparent); // background colour
|
2011-02-14 11:19:18 +01:00
|
|
|
pal.setColor(QPalette::Text, QColor(255, 255, 255)); // text colour
|
|
|
|
legend->setPalette(pal);
|
|
|
|
|
|
|
|
insertLegend(legend, QwtPlot::TopLegend);
|
2011-02-27 11:26:50 +01:00
|
|
|
// insertLegend(legend, QwtPlot::ExternalLegend);
|
2011-02-14 11:19:18 +01:00
|
|
|
|
|
|
|
// // Show a legend at the bottom
|
|
|
|
// QwtLegend *legend = new QwtLegend();
|
|
|
|
// legend->setItemMode(QwtLegend::CheckableItem);
|
|
|
|
// legend->setFrameStyle(QFrame::Box | QFrame::Sunken);
|
|
|
|
// insertLegend(legend, QwtPlot::BottomLegend);
|
|
|
|
|
2011-03-13 15:14:52 +01:00
|
|
|
// Update the checked/unchecked state of the legend items
|
|
|
|
// -> this is necessary when hiding a legend where some plots are
|
|
|
|
// not visible, and the 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);
|
|
|
|
}
|
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
connect(this, SIGNAL(legendChecked(QwtPlotItem *, bool)), this, SLOT(showCurve(QwtPlotItem *, bool)));
|
|
|
|
}
|
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
void ScopeGadgetWidget::preparePlot(PlotType plotType)
|
|
|
|
{
|
|
|
|
m_plotType = plotType;
|
|
|
|
|
|
|
|
clearCurvePlots();
|
|
|
|
|
|
|
|
setMinimumSize(64, 64);
|
|
|
|
setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
|
2011-02-13 22:04:31 +01:00
|
|
|
|
|
|
|
setMargin(1);
|
|
|
|
|
|
|
|
// QPalette pal = palette();
|
|
|
|
// QPalette::ColorRole cr = backgroundRole();
|
|
|
|
// pal.setColor(cr, QColor(128, 128, 128)); // background colour
|
|
|
|
// pal.setColor(QPalette::Text, QColor(255, 255, 255)); // text colour
|
|
|
|
// setPalette(pal);
|
2010-08-14 22:25:55 +02:00
|
|
|
|
2011-02-13 11:11:51 +01:00
|
|
|
// setCanvasBackground(Utils::StyleHelper::baseColor());
|
2011-02-13 22:04:31 +01:00
|
|
|
setCanvasBackground(QColor(64, 64, 64));
|
2010-08-14 22:25:55 +02:00
|
|
|
|
|
|
|
//Add grid lines
|
|
|
|
QwtPlotGrid *grid = new QwtPlotGrid;
|
2011-02-13 11:11:51 +01:00
|
|
|
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);
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
// Add the legend
|
|
|
|
addLegend();
|
2010-08-14 22:25:55 +02:00
|
|
|
|
2011-01-20 23:30:49 +01:00
|
|
|
// Only start the timer if we are already connected
|
|
|
|
Core::ConnectionManager *cm = Core::ICore::instance()->connectionManager();
|
2011-02-27 11:26:50 +01:00
|
|
|
if (cm->getCurrentConnection() && replotTimer)
|
|
|
|
{
|
|
|
|
if (!replotTimer->isActive())
|
|
|
|
replotTimer->start(m_refreshInterval);
|
|
|
|
else
|
|
|
|
replotTimer->setInterval(m_refreshInterval);
|
|
|
|
}
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::showCurve(QwtPlotItem *item, bool on)
|
|
|
|
{
|
|
|
|
item->setVisible(!on);
|
|
|
|
QWidget *w = legend()->find(item);
|
|
|
|
if ( w && w->inherits("QwtLegendItem") )
|
|
|
|
((QwtLegendItem *)w)->setChecked(on);
|
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
mutex.lock();
|
|
|
|
replot();
|
|
|
|
mutex.unlock();
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::setupSequencialPlot()
|
|
|
|
{
|
|
|
|
preparePlot(SequencialPlot);
|
|
|
|
|
2011-02-13 22:04:31 +01:00
|
|
|
// QwtText title("Index");
|
|
|
|
//// title.setFont(QFont("Helvetica", 20));
|
|
|
|
// title.font().setPointSize(title.font().pointSize() / 2);
|
|
|
|
// setAxisTitle(QwtPlot::xBottom, title);
|
|
|
|
//// setAxisTitle(QwtPlot::xBottom, "Index");
|
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
setAxisScaleDraw(QwtPlot::xBottom, new QwtScaleDraw());
|
|
|
|
setAxisScale(QwtPlot::xBottom, 0, m_xWindowSize);
|
|
|
|
setAxisLabelRotation(QwtPlot::xBottom, 0.0);
|
|
|
|
setAxisLabelAlignment(QwtPlot::xBottom, Qt::AlignLeft | Qt::AlignBottom);
|
2011-02-14 01:44:33 +01:00
|
|
|
|
|
|
|
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
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::setupChronoPlot()
|
|
|
|
{
|
|
|
|
preparePlot(ChronoPlot);
|
|
|
|
|
2011-02-13 22:04:31 +01:00
|
|
|
// QwtText title("Time [h:m:s]");
|
|
|
|
//// title.setFont(QFont("Helvetica", 20));
|
|
|
|
// title.font().setPointSize(title.font().pointSize() / 2);
|
|
|
|
// setAxisTitle(QwtPlot::xBottom, title);
|
|
|
|
//// setAxisTitle(QwtPlot::xBottom, "Time [h:m:s]");
|
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
setAxisScaleDraw(QwtPlot::xBottom, new TimeScaleDraw());
|
|
|
|
uint NOW = QDateTime::currentDateTime().toTime_t();
|
2010-08-14 22:25:55 +02:00
|
|
|
setAxisScale(QwtPlot::xBottom, NOW - m_xWindowSize / 1000, NOW);
|
2011-02-14 01:44:33 +01:00
|
|
|
// setAxisLabelRotation(QwtPlot::xBottom, -15.0);
|
|
|
|
setAxisLabelRotation(QwtPlot::xBottom, 0.0);
|
|
|
|
setAxisLabelAlignment(QwtPlot::xBottom, Qt::AlignLeft | Qt::AlignBottom);
|
|
|
|
// setAxisLabelAlignment(QwtPlot::xBottom, Qt::AlignCenter | Qt::AlignBottom);
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2011-02-13 22:04:31 +01:00
|
|
|
QwtScaleWidget *scaleWidget = axisWidget(QwtPlot::xBottom);
|
|
|
|
// QwtScaleDraw *scaleDraw = axisScaleDraw();
|
|
|
|
|
2011-02-14 01:44:33 +01:00
|
|
|
// 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
|
|
|
|
|
|
|
|
// set the axis colours .. can't seem to change the background colour :(
|
2011-02-13 22:04:31 +01:00
|
|
|
// QPalette pal = scaleWidget->palette();
|
|
|
|
// QPalette::ColorRole cr = scaleWidget->backgroundRole();
|
|
|
|
// pal.setColor(cr, QColor(128, 128, 128)); // background colour
|
|
|
|
// cr = scaleWidget->foregroundRole();
|
|
|
|
// pal.setColor(cr, QColor(255, 255, 255)); // tick colour
|
|
|
|
// pal.setColor(QPalette::Text, QColor(255, 255, 255)); // text colour
|
|
|
|
// scaleWidget->setPalette(pal);
|
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
/*
|
|
|
|
In situations, when there is a label at the most right position of the
|
|
|
|
scale, additional space is needed to display the overlapping part
|
|
|
|
of the label would be taken by reducing the width of scale and canvas.
|
|
|
|
To avoid this "jumping canvas" effect, we add a permanent margin.
|
|
|
|
We don't need to do the same for the left border, because there
|
|
|
|
is enough space for the overlapping label below the left scale.
|
|
|
|
*/
|
|
|
|
|
2011-02-14 01:44:33 +01:00
|
|
|
// const int fmh = QFontMetrics(scaleWidget->font()).height();
|
|
|
|
// scaleWidget->setMinBorderDist(0, fmh / 2);
|
|
|
|
|
|
|
|
// const int fmw = QFontMetrics(scaleWidget->font()).width(" 00:00:00 ");
|
|
|
|
// const int fmw = QFontMetrics(scaleWidget->font()).width(" ");
|
|
|
|
// scaleWidget->setMinBorderDist(0, fmw);
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
2011-11-18 18:30:58 +01:00
|
|
|
void ScopeGadgetWidget::addCurvePlot(QString uavObject, QString uavFieldSubField, int scaleOrderFactor, int interpolationSamples, QPen pen)
|
2010-08-10 07:42:39 +02:00
|
|
|
{
|
|
|
|
PlotData* plotData;
|
|
|
|
|
|
|
|
if (m_plotType == SequencialPlot)
|
2010-09-01 23:29:24 +02:00
|
|
|
plotData = new SequencialPlotData(uavObject, uavFieldSubField);
|
2010-08-10 07:42:39 +02:00
|
|
|
else if (m_plotType == ChronoPlot)
|
2010-09-01 23:29:24 +02:00
|
|
|
plotData = new ChronoPlotData(uavObject, uavFieldSubField);
|
2010-08-10 07:42:39 +02:00
|
|
|
//else if (m_plotType == UAVObjectPlot)
|
|
|
|
// plotData = new UAVObjectPlotData(uavObject, uavField);
|
|
|
|
|
|
|
|
plotData->m_xWindowSize = m_xWindowSize;
|
|
|
|
plotData->scalePower = scaleOrderFactor;
|
2011-11-18 18:30:58 +01:00
|
|
|
plotData->interpolationSamples = interpolationSamples;
|
2010-08-10 07:42:39 +02:00
|
|
|
|
|
|
|
//If the y-bounds are supplied, set them
|
|
|
|
if (plotData->yMinimum != plotData->yMaximum)
|
2011-02-14 01:44:33 +01:00
|
|
|
{
|
2011-02-13 22:04:31 +01:00
|
|
|
setAxisScale(QwtPlot::yLeft, plotData->yMinimum, plotData->yMaximum);
|
2011-02-14 01:44:33 +01:00
|
|
|
}
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2010-08-30 16:17:15 +02:00
|
|
|
//Create the curve
|
|
|
|
QString curveName = (plotData->uavObject) + "." + (plotData->uavField);
|
|
|
|
if(plotData->haveSubField)
|
|
|
|
curveName = curveName.append("." + plotData->uavSubField);
|
|
|
|
|
2010-09-01 23:29:24 +02:00
|
|
|
//Get the uav object
|
|
|
|
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
|
|
|
|
UAVObjectManager *objManager = pm->getObject<UAVObjectManager>();
|
|
|
|
UAVDataObject* obj = dynamic_cast<UAVDataObject*>(objManager->getObject((plotData->uavObject)));
|
2011-08-29 18:37:22 +02:00
|
|
|
if(!obj) {
|
|
|
|
qDebug() << "Object " << plotData->uavObject << " is missing";
|
|
|
|
return;
|
|
|
|
}
|
2010-09-01 23:29:24 +02:00
|
|
|
UAVObjectField* field = obj->getField(plotData->uavField);
|
2011-08-29 18:37:22 +02:00
|
|
|
if(!field) {
|
|
|
|
qDebug() << "Field " << plotData->uavField << " of object " << plotData->uavObject << " is missing";
|
|
|
|
return;
|
|
|
|
}
|
2010-09-01 23:29:24 +02:00
|
|
|
QString units = field->getUnits();
|
|
|
|
|
|
|
|
if(units == 0)
|
|
|
|
units = QString();
|
|
|
|
|
2010-08-30 16:17:15 +02:00
|
|
|
QString curveNameScaled;
|
2010-08-30 00:45:12 +02:00
|
|
|
if(scaleOrderFactor == 0)
|
2010-09-01 23:29:24 +02:00
|
|
|
curveNameScaled = curveName + "(" + units + ")";
|
2010-08-30 00:45:12 +02:00
|
|
|
else
|
2010-09-01 23:29:24 +02:00
|
|
|
curveNameScaled = curveName + "(x10^" + QString::number(scaleOrderFactor) + " " + units + ")";
|
2010-08-30 00:45:12 +02:00
|
|
|
|
2010-08-30 16:17:15 +02:00
|
|
|
QwtPlotCurve* plotCurve = new QwtPlotCurve(curveNameScaled);
|
2010-08-10 07:42:39 +02:00
|
|
|
plotCurve->setPen(pen);
|
|
|
|
plotCurve->setData(*plotData->xData, *plotData->yData);
|
|
|
|
plotCurve->attach(this);
|
|
|
|
plotData->curve = plotCurve;
|
|
|
|
|
|
|
|
//Keep the curve details for later
|
2010-08-30 16:17:15 +02:00
|
|
|
m_curvesData.insert(curveNameScaled, plotData);
|
2010-08-10 07:42:39 +02:00
|
|
|
|
|
|
|
//Link to the signal of new data only if this UAVObject has not been to connected yet
|
|
|
|
if (!m_connectedUAVObjects.contains(obj->getName())) {
|
|
|
|
m_connectedUAVObjects.append(obj->getName());
|
|
|
|
connect(obj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(uavObjectReceived(UAVObject*)));
|
|
|
|
}
|
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
mutex.lock();
|
|
|
|
replot();
|
|
|
|
mutex.unlock();
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
2010-08-30 16:17:15 +02:00
|
|
|
//void ScopeGadgetWidget::removeCurvePlot(QString uavObject, QString uavField)
|
|
|
|
//{
|
|
|
|
// QString curveName = uavObject + "." + uavField;
|
|
|
|
//
|
|
|
|
// PlotData* plotData = m_curvesData.take(curveName);
|
|
|
|
// m_curvesData.remove(curveName);
|
|
|
|
// plotData->curve->detach();
|
|
|
|
//
|
|
|
|
// delete plotData->curve;
|
|
|
|
// delete plotData;
|
|
|
|
//
|
2011-02-14 11:19:18 +01:00
|
|
|
// mutex.lock();
|
|
|
|
// replot();
|
|
|
|
// mutex.unlock();
|
2010-08-30 16:17:15 +02:00
|
|
|
//}
|
2010-08-10 07:42:39 +02:00
|
|
|
|
|
|
|
void ScopeGadgetWidget::uavObjectReceived(UAVObject* obj)
|
|
|
|
{
|
|
|
|
foreach(PlotData* plotData, m_curvesData.values()) {
|
2011-02-01 18:43:30 +01:00
|
|
|
if (plotData->append(obj)) m_csvLoggingDataUpdated=1;
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::replotNewData()
|
|
|
|
{
|
2011-02-14 11:19:18 +01:00
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
|
|
|
|
foreach(PlotData* plotData, m_curvesData.values())
|
|
|
|
{
|
2010-08-14 22:25:55 +02:00
|
|
|
plotData->removeStaleData();
|
|
|
|
plotData->curve->setData(*plotData->xData, *plotData->yData);
|
|
|
|
}
|
|
|
|
|
|
|
|
QDateTime NOW = QDateTime::currentDateTime();
|
|
|
|
double toTime = NOW.toTime_t();
|
|
|
|
toTime += NOW.time().msec() / 1000.0;
|
2011-02-14 11:19:18 +01:00
|
|
|
if (m_plotType == ChronoPlot)
|
2010-08-14 22:25:55 +02:00
|
|
|
setAxisScale(QwtPlot::xBottom, toTime - m_xWindowSize, toTime);
|
2011-02-14 11:19:18 +01:00
|
|
|
|
|
|
|
// qDebug() << "replotNewData from " << NOW.addSecs(- m_xWindowSize) << " to " << NOW;
|
|
|
|
|
2011-01-31 03:09:44 +01:00
|
|
|
csvLoggingInsertData();
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2011-02-14 11:19:18 +01:00
|
|
|
replot();
|
|
|
|
}
|
2010-08-10 07:42:39 +02:00
|
|
|
|
|
|
|
void ScopeGadgetWidget::setupExamplePlot()
|
|
|
|
{
|
|
|
|
preparePlot(SequencialPlot);
|
|
|
|
|
|
|
|
// Show the axes
|
|
|
|
|
|
|
|
setAxisTitle(xBottom, "x");
|
|
|
|
setAxisTitle(yLeft, "y");
|
|
|
|
|
|
|
|
// Calculate the data, 500 points each
|
|
|
|
const int points = 500;
|
|
|
|
double x[ points ];
|
|
|
|
double sn[ points ];
|
|
|
|
double cs[ points ];
|
|
|
|
double sg[ points ];
|
|
|
|
|
|
|
|
for (int i = 0; i < points; i++) {
|
|
|
|
x[i] = (3.0 * 3.14 / double(points)) * double(i);
|
|
|
|
sn[i] = 2.0 * sin(x[i]);
|
|
|
|
cs[i] = 3.0 * cos(x[i]);
|
|
|
|
sg[i] = (sn[i] > 0) ? 1 : ((sn[i] < 0) ? -1 : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// add curves
|
|
|
|
QwtPlotCurve *curve1 = new QwtPlotCurve("Curve 1");
|
|
|
|
curve1->setPen(QPen(Qt::blue));
|
|
|
|
QwtPlotCurve *curve2 = new QwtPlotCurve("Curve 2");
|
|
|
|
curve2->setPen(QPen(Qt::red));
|
|
|
|
QwtPlotCurve *curve3 = new QwtPlotCurve("Curve 3");
|
|
|
|
curve3->setPen(QPen(Qt::green));
|
|
|
|
|
|
|
|
// copy the data into the curves
|
|
|
|
curve1->setData(x, sn, points);
|
|
|
|
curve2->setData(x, cs, points);
|
|
|
|
curve3->setData(x, sg, points);
|
|
|
|
|
|
|
|
curve1->attach(this);
|
|
|
|
curve2->attach(this);
|
|
|
|
curve3->attach(this);
|
|
|
|
|
|
|
|
// finally, refresh the plot
|
2011-02-14 11:19:18 +01:00
|
|
|
mutex.lock();
|
|
|
|
replot();
|
|
|
|
mutex.unlock();
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::clearCurvePlots()
|
|
|
|
{
|
|
|
|
foreach(PlotData* plotData, m_curvesData.values()) {
|
|
|
|
plotData->curve->detach();
|
|
|
|
|
|
|
|
delete plotData->curve;
|
|
|
|
delete plotData;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_curvesData.clear();
|
|
|
|
}
|
|
|
|
|
2010-08-14 22:25:55 +02:00
|
|
|
|
2010-08-10 07:42:39 +02:00
|
|
|
TestDataGen::TestDataGen()
|
|
|
|
{
|
|
|
|
// Get required UAVObjects
|
|
|
|
ExtensionSystem::PluginManager* pm = ExtensionSystem::PluginManager::instance();
|
|
|
|
UAVObjectManager* objManager = pm->getObject<UAVObjectManager>();
|
|
|
|
|
2010-08-21 18:19:10 +02:00
|
|
|
baroAltitude = BaroAltitude::GetInstance(objManager);
|
2010-08-10 07:42:39 +02:00
|
|
|
gps = PositionActual::GetInstance(objManager);
|
2010-08-30 16:17:15 +02:00
|
|
|
attRaw = AttitudeRaw::GetInstance(objManager);
|
|
|
|
manCtrlCmd = ManualControlCommand::GetInstance(objManager);
|
2010-08-10 07:42:39 +02:00
|
|
|
|
|
|
|
//Setup timer
|
2010-08-30 16:17:15 +02:00
|
|
|
periodMs = 20;
|
2010-08-10 07:42:39 +02:00
|
|
|
timer = new QTimer(this);
|
|
|
|
connect(timer, SIGNAL(timeout()), this, SLOT(genTestData()));
|
2010-08-14 22:25:55 +02:00
|
|
|
timer->start(periodMs);
|
|
|
|
|
|
|
|
debugCounter = 0;
|
2010-08-30 16:17:15 +02:00
|
|
|
testTime = 0;
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void TestDataGen::genTestData()
|
|
|
|
{
|
2010-08-21 18:19:10 +02:00
|
|
|
// Update BaroAltitude object
|
|
|
|
BaroAltitude::DataFields baroAltitudeData;
|
|
|
|
baroAltitudeData.Altitude = 500 * sin(1 * testTime) + 200 * cos(4 * testTime) + 800;
|
|
|
|
baroAltitudeData.Temperature = 30 * sin(0.5 * testTime);
|
2010-08-30 00:45:12 +02:00
|
|
|
baroAltitudeData.Pressure = baroAltitudeData.Altitude * 0.01 + baroAltitudeData.Temperature;
|
2010-08-21 18:19:10 +02:00
|
|
|
baroAltitude->setData(baroAltitudeData);
|
2010-08-10 07:42:39 +02:00
|
|
|
|
2010-08-30 16:17:15 +02:00
|
|
|
// Update Attitude Raw data
|
|
|
|
AttitudeRaw::DataFields attData;
|
2010-08-30 21:39:56 +02:00
|
|
|
// attData.accels[0] = 4 * sin(2 * testTime) + 1 * cos(6 * testTime) + 4;
|
2010-08-30 16:17:15 +02:00
|
|
|
// attData.accels[1] = 3 * sin(2.3 * testTime) + 1.5 * cos(3.3 * testTime) + 2;
|
|
|
|
// attData.accels[2] = 4 * sin(5.3 * testTime) + 1.5 * cos(1.3 * testTime) + 1;
|
|
|
|
attData.accels[0] = 1;
|
|
|
|
attData.accels[1] = 4;
|
|
|
|
attData.accels[2] = 9;
|
|
|
|
attRaw->setData(attData);
|
|
|
|
|
|
|
|
|
|
|
|
ManualControlCommand::DataFields manCtlData;
|
|
|
|
manCtlData.Channel[0] = 400 * cos(2 * testTime) + 100 * sin(6 * testTime) + 400;
|
|
|
|
manCtlData.Channel[1] = 350 * cos(2.3 * testTime) + 150 * sin(3.3 * testTime) + 200;
|
|
|
|
manCtlData.Channel[2] = 450 * cos(5.3 * testTime) + 150 * sin(1.3 * testTime) + 150;
|
|
|
|
manCtrlCmd->setData(manCtlData);
|
|
|
|
|
2010-08-14 22:25:55 +02:00
|
|
|
testTime += (periodMs / 1000.0);
|
|
|
|
|
2010-08-26 21:54:23 +02:00
|
|
|
// debugCounter++;
|
|
|
|
// if (debugCounter % (100/periodMs) == 0 )
|
|
|
|
// qDebug() << "Test Time = " << testTime;
|
2010-08-10 07:42:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TestDataGen::~TestDataGen()
|
|
|
|
{
|
|
|
|
if (timer)
|
|
|
|
timer->stop();
|
|
|
|
|
|
|
|
delete timer;
|
|
|
|
}
|
2011-01-31 03:09:44 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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;
|
|
|
|
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 );
|
2011-02-01 18:43:30 +01:00
|
|
|
ts << "date" << ", " << "Time"<< ", " << "Sec since start"<< ", " << "Connected" << ", " << "Data changed";
|
2011-01-31 03:09:44 +01:00
|
|
|
|
|
|
|
foreach(PlotData* plotData2, m_curvesData.values())
|
|
|
|
{
|
|
|
|
ts << ", ";
|
|
|
|
ts << plotData2->uavObject;
|
|
|
|
ts << "." << plotData2->uavField;
|
|
|
|
if (plotData2->haveSubField) ts << "." << plotData2->uavSubField;
|
|
|
|
}
|
|
|
|
ts << endl;
|
|
|
|
m_csvLoggingFile.close();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ScopeGadgetWidget::csvLoggingInsertData()
|
|
|
|
{
|
|
|
|
if (!m_csvLoggingStarted) return -1;
|
|
|
|
m_csvLoggingDataSaved=1;
|
|
|
|
m_csvLoggingDataValid=0;
|
|
|
|
QDateTime NOW = QDateTime::currentDateTime();
|
|
|
|
QString tempString;
|
|
|
|
|
|
|
|
if(m_csvLoggingFile.open(QIODevice::WriteOnly | QIODevice::Append)== FALSE)
|
|
|
|
{
|
|
|
|
qDebug() << "Unable to open " << m_csvLoggingFile.fileName() << " for csv logging Data";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QTextStream ss( &tempString );
|
2011-02-01 18:43:30 +01:00
|
|
|
ss << NOW.toString("yyyy-MM-dd") << ", " << NOW.toString("hh:mm:ss.z") << ", " ;
|
|
|
|
|
2011-02-01 15:57:39 +01:00
|
|
|
#if QT_VERSION >= 0x040700
|
2011-02-01 18:43:30 +01:00
|
|
|
ss <<(NOW.toMSecsSinceEpoch() - m_csvLoggingStartTime.toMSecsSinceEpoch())/1000.00;
|
2011-02-01 15:57:39 +01:00
|
|
|
#else
|
2011-02-01 18:43:30 +01:00
|
|
|
ss <<(NOW.toTime_t() - m_csvLoggingStartTime.toTime_t());
|
2011-02-01 15:57:39 +01:00
|
|
|
#endif
|
2011-02-01 18:43:30 +01:00
|
|
|
ss << ", " << m_csvLoggingConnected << ", " << m_csvLoggingDataUpdated;
|
|
|
|
m_csvLoggingDataUpdated=0;
|
|
|
|
|
2011-01-31 03:09:44 +01:00
|
|
|
foreach(PlotData* plotData2, m_curvesData.values())
|
|
|
|
{
|
|
|
|
ss << ", ";
|
|
|
|
if (plotData2->xData->isEmpty ())
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-11-26 22:06:21 +01:00
|
|
|
ss << QString().sprintf("%3.6g",plotData2->yDataHistory->last()/pow(10,plotData2->scalePower));
|
2011-01-31 03:09:44 +01:00
|
|
|
m_csvLoggingDataValid=1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ss << endl;
|
|
|
|
if (m_csvLoggingDataValid)
|
|
|
|
{
|
|
|
|
QTextStream ts( &m_csvLoggingFile );
|
|
|
|
ts << tempString;
|
|
|
|
}
|
|
|
|
m_csvLoggingFile.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2011-02-14 11:19:18 +01:00
|
|
|
|
2011-01-31 03:09:44 +01:00
|
|
|
void ScopeGadgetWidget::csvLoggingSetName(QString newName)
|
|
|
|
{
|
|
|
|
m_csvLoggingName = newName;
|
|
|
|
m_csvLoggingNameSet=1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScopeGadgetWidget::csvLoggingConnect()
|
|
|
|
{
|
|
|
|
m_csvLoggingConnected=1;
|
|
|
|
if (m_csvLoggingNewFileOnConnect)csvLoggingStart();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
void ScopeGadgetWidget::csvLoggingDisconnect()
|
|
|
|
{
|
|
|
|
m_csvLoggingHeaderSaved=0;
|
|
|
|
m_csvLoggingConnected=0;
|
|
|
|
if (m_csvLoggingNewFileOnConnect)csvLoggingStop();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|