1
0
mirror of https://bitbucket.org/librepilot/librepilot.git synced 2025-01-25 10:52:11 +01:00
LibrePilot/ground/openpilotgcs/src/plugins/scope/scopegadgetwidget.cpp

627 lines
19 KiB
C++
Raw Normal View History

/**
******************************************************************************
*
* @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_plot_grid.h"
#include <iostream>
#include <math.h>
#include <QDebug>
#include <QDir>
#include <QColor>
#include <QStringList>
#include <QWidget>
#include <QVBoxLayout>
#include <QPushButton>
#include <QMutexLocker>
#include <QWheelEvent>
2014-10-30 01:12:43 +01:00
#include <qwt/src/qwt_legend_label.h>
#include <qwt/src/qwt_plot_canvas.h>
#include <qwt/src/qwt_plot_layout.h>
ScopeGadgetWidget::ScopeGadgetWidget(QWidget *parent) : QwtPlot(parent),
m_csvLoggingStarted(false), m_csvLoggingEnabled(false),
m_csvLoggingHeaderSaved(false), m_csvLoggingDataSaved(false),
m_csvLoggingNameSet(false), m_csvLoggingDataValid(false),
m_csvLoggingDataUpdated(false), m_csvLoggingConnected(false),
m_csvLoggingNewFileOnConnect(false),
m_csvLoggingStartTime(QDateTime::currentDateTime()),
m_csvLoggingPath("./csvlogging/"),
m_plotLegend(NULL)
{
setMouseTracking(true);
2014-10-30 12:15:37 +01:00
2014-11-03 23:09:26 +01:00
QwtPlotCanvas *plotCanvas = dynamic_cast<QwtPlotCanvas *>(canvas());
2014-10-30 01:12:43 +01:00
if (plotCanvas) {
2014-10-30 12:15:37 +01:00
plotCanvas->setFrameStyle(QFrame::StyledPanel | QFrame::Sunken);
plotCanvas->setBorderRadius(8);
2014-10-30 01:12:43 +01:00
}
axisWidget(QwtPlot::yLeft)->setMargin(2);
axisWidget(QwtPlot::xBottom)->setMargin(2);
// 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()));
// 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<UAVObjectManager>();
foreach(QString uavObjName, m_connectedUAVObjects) {
UAVDataObject *obj = dynamic_cast<UAVDataObject *>(objManager->getObject(uavObjName));
2014-11-03 23:09:26 +01:00
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
2014-10-30 01:12:43 +01:00
m_mutex.lock();
if (legend()) {
deleteLegend();
} else {
addLegend();
}
2014-10-30 01:12:43 +01:00
m_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
2014-10-30 01:12:43 +01:00
m_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);
}
2014-10-30 01:12:43 +01:00
m_mutex.unlock();
}
QwtPlot::wheelEvent(e);
}
void ScopeGadgetWidget::showEvent(QShowEvent *e)
{
replotNewData();
QwtPlot::showEvent(e);
}
/**
* Starts/stops telemetry
*/
void ScopeGadgetWidget::startPlotting()
{
2014-10-28 18:12:04 +01:00
if (replotTimer && !replotTimer->isActive()) {
replotTimer->start(m_refreshInterval);
}
}
void ScopeGadgetWidget::stopPlotting()
{
2014-10-28 18:12:04 +01:00
if (replotTimer) {
replotTimer->stop();
}
}
void ScopeGadgetWidget::deleteLegend()
{
2014-10-30 01:12:43 +01:00
if (m_plotLegend) {
2014-10-28 18:12:04 +01:00
insertLegend(NULL, QwtPlot::TopLegend);
2014-10-30 01:12:43 +01:00
m_plotLegend = NULL;
}
}
void ScopeGadgetWidget::addLegend()
{
if (legend()) {
return;
}
// Show a legend at the top
2014-10-30 01:12:43 +01:00
m_plotLegend = new QwtLegend(this);
m_plotLegend->setDefaultItemMode(QwtLegendData::Checkable);
m_plotLegend->setFrameStyle(QFrame::NoFrame);
m_plotLegend->setToolTip(tr("Click legend to show/hide scope trace.\nDouble click legend or plot to show/hide legend."));
// set colors
2014-10-30 01:12:43 +01:00
QPalette pal = m_plotLegend->palette();
pal.setColor(m_plotLegend->backgroundRole(), QColor(100, 100, 100));
pal.setColor(QPalette::Text, QColor(0, 0, 0));
2014-10-30 01:12:43 +01:00
m_plotLegend->setPalette(pal);
2014-10-30 01:12:43 +01:00
insertLegend(m_plotLegend, 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.
2014-10-30 01:12:43 +01:00
foreach(QwtPlotItem * plotItem, itemList()) {
QWidget *legendWidget = m_plotLegend->legendWidget(QwtPlot::itemToInfo(plotItem));
2014-11-03 23:09:26 +01:00
2014-10-30 01:12:43 +01:00
if (legendWidget && legendWidget->inherits("QwtLegendLabel")) {
((QwtLegendLabel *)legendWidget)->setChecked(!plotItem->isVisible());
}
}
2014-11-03 23:09:26 +01:00
connect(m_plotLegend, SIGNAL(checked(QVariant, bool, int)), this, SLOT(showCurve(QVariant, bool, int)));
}
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;
2014-10-30 12:15:37 +01:00
grid->setPen(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);
}
}
}
2014-10-30 01:12:43 +01:00
void ScopeGadgetWidget::showCurve(QVariant itemInfo, bool visible, int index)
{
2014-10-30 01:12:43 +01:00
Q_UNUSED(index);
QwtPlotItem *item = infoToItem(itemInfo);
item->setVisible(!visible);
emit visibilityChanged(item);
2014-10-30 01:12:43 +01:00
if (m_plotLegend) {
QWidget *legendItem = legend()->find((WId)item);
if (legendItem && legendItem->inherits("QwtLegendLabel")) {
((QwtLegendLabel *)legendItem)->setChecked(visible);
}
}
2014-10-30 01:12:43 +01:00
m_mutex.lock();
replot();
2014-10-30 01:12:43 +01:00
m_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);
// 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);
// 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
}
2014-10-28 13:33:21 +01:00
void ScopeGadgetWidget::addCurvePlot(QString objectName, QString fieldPlusSubField, int scaleFactor,
int meanSamples, QString mathFunction, QPen pen, bool antialiased)
{
QString fieldName = fieldPlusSubField;
QString elementName;
2014-10-28 13:33:21 +01:00
int element = 0;
if (fieldPlusSubField.contains("-")) {
QStringList fieldSubfield = fieldName.split("-", QString::SkipEmptyParts);
2014-11-03 23:09:26 +01:00
fieldName = fieldSubfield.at(0);
elementName = fieldSubfield.at(1);
}
// Get the uav object
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
UAVObjectManager *objManager = pm->getObject<UAVObjectManager>();
2014-10-28 13:33:21 +01:00
UAVDataObject *object = dynamic_cast<UAVDataObject *>(objManager->getObject(objectName));
if (!object) {
qDebug() << "Object" << objectName << "is missing";
return;
}
2014-10-28 13:33:21 +01:00
UAVObjectField *field = object->getField(fieldName);
if (!field) {
qDebug() << "In scope gadget, in fields loaded from GCS config file, field" <<
2014-11-03 23:09:26 +01:00
fieldName << "of object" << objectName << "is missing";
return;
}
2014-10-28 13:33:21 +01:00
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" <<
2014-11-03 23:09:26 +01:00
fieldName << "of object" << objectName << "element name" << elementName << "is missing";
2014-10-28 13:33:21 +01:00
return;
}
}
PlotData *plotData;
if (m_plotType == SequentialPlot) {
plotData = new SequentialPlotData(object, field, element, scaleFactor,
meanSamples, mathFunction, m_plotDataSize,
pen, antialiased);
} else if (m_plotType == ChronoPlot) {
plotData = new ChronoPlotData(object, field, element, scaleFactor,
meanSamples, mathFunction, m_plotDataSize,
2014-11-03 23:09:26 +01:00
pen, antialiased);
}
2014-11-03 23:09:26 +01:00
connect(this, SIGNAL(visibilityChanged(QwtPlotItem *)), plotData, SLOT(visibilityChanged(QwtPlotItem *)));
plotData->attach(this);
if (plotData->wantsInitialData()) {
plotData->append(object);
}
// Keep the curve details for later
2014-10-29 00:05:53 +01:00
m_curvesData.insert(plotData->plotName(), plotData);
// Link to the new signal data only if this UAVObject has not been connected yet
2014-10-28 13:33:21 +01:00
if (!m_connectedUAVObjects.contains(object->getName())) {
m_connectedUAVObjects.append(object->getName());
connect(object, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(uavObjectReceived(UAVObject *)));
}
2014-10-30 01:12:43 +01:00
m_mutex.lock();
replot();
2014-10-30 01:12:43 +01:00
m_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;
}
2014-10-30 01:12:43 +01:00
QMutexLocker locker(&m_mutex);
foreach(PlotData * plotData, m_curvesData.values()) {
plotData->removeStaleData();
2014-10-29 00:05:53 +01:00
plotData->updatePlotData();
}
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();
2014-03-20 01:46:38 +01:00
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()) {
2014-10-29 00:05:53 +01:00
plotData->setVisible(qSettings->value(QString("plot%1").arg(i), true).toBool());
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();
2014-10-28 13:33:21 +01:00
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;
}
2014-10-28 12:19:54 +01:00
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;
2014-10-28 12:19:54 +01:00
m_csvLoggingDataUpdated = false;
foreach(PlotData * plotData2, m_curvesData.values()) {
ss << ", ";
2014-10-28 12:19:54 +01:00
if (plotData2->hasData()) {
ss << plotData2->lastDataAsString();
2014-10-28 12:19:54 +01:00
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();
}
}