/*===================================================================== QGroundControl Open Source Ground Control Station (c) 2009, 2010 QGROUNDCONTROL PROJECT This file is part of the QGROUNDCONTROL project QGROUNDCONTROL 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. QGROUNDCONTROL 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 QGROUNDCONTROL. If not, see . ======================================================================*/ /** * @file * @brief Implementation of QGCDataPlot2D * @author Lorenz Meier * */ #include #include #include #include #include #include #include #include #include #include "QGCDataPlot2D.h" #include "ui_QGCDataPlot2D.h" #include "MG.h" #include #include QGCDataPlot2D::QGCDataPlot2D(QWidget *parent) : QWidget(parent), plot(new IncrementalPlot()), logFile(NULL), ui(new Ui::QGCDataPlot2D) { ui->setupUi(this); // Add plot to ui QHBoxLayout* layout = new QHBoxLayout(ui->plotFrame); layout->addWidget(plot); ui->plotFrame->setLayout(layout); ui->gridCheckBox->setChecked(plot->gridEnabled()); // Connect user actions connect(ui->selectFileButton, SIGNAL(clicked()), this, SLOT(selectFile())); connect(ui->saveCsvButton, SIGNAL(clicked()), this, SLOT(saveCsvLog())); connect(ui->reloadButton, SIGNAL(clicked()), this, SLOT(reloadFile())); connect(ui->savePlotButton, SIGNAL(clicked()), this, SLOT(savePlot())); connect(ui->printButton, SIGNAL(clicked()), this, SLOT(print())); connect(ui->legendCheckBox, SIGNAL(clicked(bool)), plot, SLOT(showLegend(bool))); connect(ui->symmetricCheckBox, SIGNAL(clicked(bool)), plot, SLOT(setSymmetric(bool))); connect(ui->gridCheckBox, SIGNAL(clicked(bool)), plot, SLOT(showGrid(bool))); connect(ui->regressionButton, SIGNAL(clicked()), this, SLOT(calculateRegression())); connect(ui->style, SIGNAL(currentIndexChanged(QString)), plot, SLOT(setStyleText(QString))); } void QGCDataPlot2D::reloadFile() { if (QFileInfo(fileName).isReadable()) { if (ui->inputFileType->currentText().contains("pxIMU")) { loadRawLog(fileName, ui->xAxis->currentText(), ui->yAxis->text()); } else if (ui->inputFileType->currentText().contains("CSV")) { loadCsvLog(fileName, ui->xAxis->currentText(), ui->yAxis->text()); } } } void QGCDataPlot2D::loadFile() { if (QFileInfo(fileName).isReadable()) { if (ui->inputFileType->currentText().contains("pxIMU")) { loadRawLog(fileName); } else if (ui->inputFileType->currentText().contains("CSV")) { loadCsvLog(fileName); } } } void QGCDataPlot2D::loadFile(QString file) { fileName = file; if (QFileInfo(fileName).isReadable()) { if (fileName.contains(".raw") || fileName.contains(".imu")) { loadRawLog(fileName); } else if (fileName.contains(".txt") || fileName.contains(".csv")) { loadCsvLog(fileName); } } } /** * This function brings up a file name dialog and exports to either PDF or SVG, depending on the filename */ void QGCDataPlot2D::savePlot() { QString fileName = "plot.svg"; fileName = QFileDialog::getSaveFileName( this, "Export File Name", QDesktopServices::storageLocation(QDesktopServices::DesktopLocation), "PDF Documents (*.pdf);;SVG Images (*.svg)"); if (!fileName.contains(".")) { // .csv is default extension fileName.append(".pdf"); } while(!(fileName.endsWith(".svg") || fileName.endsWith(".pdf"))) { QMessageBox msgBox; msgBox.setIcon(QMessageBox::Critical); msgBox.setText("Unsuitable file extension for PDF or SVG"); msgBox.setInformativeText("Please choose .pdf or .svg as file extension. Click OK to change the file extension, cancel to not save the file."); msgBox.setStandardButtons(QMessageBox::Ok | QMessageBox::Cancel); msgBox.setDefaultButton(QMessageBox::Ok); // Abort if cancelled if(msgBox.exec() == QMessageBox::Cancel) return; fileName = QFileDialog::getSaveFileName( this, "Export File Name", QDesktopServices::storageLocation(QDesktopServices::DesktopLocation), "PDF Documents (*.pdf);;SVG Images (*.svg)"); } if (fileName.endsWith(".pdf")) { exportPDF(fileName); } else if (fileName.endsWith(".svg")) { exportSVG(fileName); } } void QGCDataPlot2D::print() { QPrinter printer(QPrinter::HighResolution); // printer.setOutputFormat(QPrinter::PdfFormat); // //QPrinter printer(QPrinter::HighResolution); // printer.setOutputFileName(fileName); QString docName = plot->title().text(); if ( !docName.isEmpty() ) { docName.replace (QRegExp (QString::fromLatin1 ("\n")), tr (" -- ")); printer.setDocName (docName); } printer.setCreator("QGroundControl"); printer.setOrientation(QPrinter::Landscape); QPrintDialog dialog(&printer); if ( dialog.exec() ) { plot->setStyleSheet("QWidget { background-color: #FFFFFF; color: #000000; background-clip: border; font-size: 10pt;}"); plot->setCanvasBackground(Qt::white); QwtPlotPrintFilter filter; filter.color(Qt::white, QwtPlotPrintFilter::CanvasBackground); filter.color(Qt::black, QwtPlotPrintFilter::AxisScale); filter.color(Qt::black, QwtPlotPrintFilter::AxisTitle); filter.color(Qt::black, QwtPlotPrintFilter::MajorGrid); filter.color(Qt::black, QwtPlotPrintFilter::MinorGrid); if ( printer.colorMode() == QPrinter::GrayScale ) { int options = QwtPlotPrintFilter::PrintAll; options &= ~QwtPlotPrintFilter::PrintBackground; options |= QwtPlotPrintFilter::PrintFrameWithScales; filter.setOptions(options); } plot->print(printer, filter); plot->setStyleSheet("QWidget { background-color: #050508; color: #DDDDDF; background-clip: border; font-size: 11pt;}"); //plot->setCanvasBackground(QColor(5, 5, 8)); } } void QGCDataPlot2D::exportPDF(QString fileName) { QPrinter printer; printer.setOutputFormat(QPrinter::PdfFormat); printer.setOutputFileName(fileName); //printer.setFullPage(true); printer.setPageMargins(10.0, 10.0, 10.0, 10.0, QPrinter::Millimeter); printer.setPageSize(QPrinter::A4); QString docName = plot->title().text(); if ( !docName.isEmpty() ) { docName.replace (QRegExp (QString::fromLatin1 ("\n")), tr (" -- ")); printer.setDocName (docName); } printer.setCreator("QGroundControl"); printer.setOrientation(QPrinter::Landscape); plot->setStyleSheet("QWidget { background-color: #FFFFFF; color: #000000; background-clip: border; font-size: 10pt;}"); // plot->setCanvasBackground(Qt::white); // QwtPlotPrintFilter filter; // filter.color(Qt::white, QwtPlotPrintFilter::CanvasBackground); // filter.color(Qt::black, QwtPlotPrintFilter::AxisScale); // filter.color(Qt::black, QwtPlotPrintFilter::AxisTitle); // filter.color(Qt::black, QwtPlotPrintFilter::MajorGrid); // filter.color(Qt::black, QwtPlotPrintFilter::MinorGrid); // if ( printer.colorMode() == QPrinter::GrayScale ) // { // int options = QwtPlotPrintFilter::PrintAll; // options &= ~QwtPlotPrintFilter::PrintBackground; // options |= QwtPlotPrintFilter::PrintFrameWithScales; // filter.setOptions(options); // } plot->print(printer);//, filter); plot->setStyleSheet("QWidget { background-color: #050508; color: #DDDDDF; background-clip: border; font-size: 11pt;}"); //plot->setCanvasBackground(QColor(5, 5, 8)); } void QGCDataPlot2D::exportSVG(QString fileName) { if ( !fileName.isEmpty() ) { plot->setStyleSheet("QWidget { background-color: #FFFFFF; color: #000000; background-clip: border; font-size: 10pt;}"); //plot->setCanvasBackground(Qt::white); QSvgGenerator generator; generator.setFileName(fileName); generator.setSize(QSize(800, 600)); QwtPlotPrintFilter filter; filter.color(Qt::white, QwtPlotPrintFilter::CanvasBackground); filter.color(Qt::black, QwtPlotPrintFilter::AxisScale); filter.color(Qt::black, QwtPlotPrintFilter::AxisTitle); filter.color(Qt::black, QwtPlotPrintFilter::MajorGrid); filter.color(Qt::black, QwtPlotPrintFilter::MinorGrid); plot->print(generator, filter); plot->setStyleSheet("QWidget { background-color: #050508; color: #DDDDDF; background-clip: border; font-size: 11pt;}"); } } /** * Selects a filename and attempts immediately to load it. */ void QGCDataPlot2D::selectFile() { // Let user select the log file name //QDate date(QDate::currentDate()); // QString("./pixhawk-log-" + date.toString("yyyy-MM-dd") + "-" + QString::number(logindex) + ".log") fileName = QFileDialog::getOpenFileName(this, tr("Specify log file name"), tr("."), tr("Logfile (*.txt)")); // Store reference to file QFileInfo fileInfo(fileName); if (!fileInfo.isReadable()) { QMessageBox msgBox; msgBox.setIcon(QMessageBox::Critical); msgBox.setText("Could not open file"); msgBox.setInformativeText(tr("The file is owned by user %1. Is the file currently used by another program?").arg(fileInfo.owner())); msgBox.setStandardButtons(QMessageBox::Ok); msgBox.setDefaultButton(QMessageBox::Ok); msgBox.exec(); ui->filenameLabel->setText(tr("Could not open %1").arg(fileInfo.baseName()+"."+fileInfo.completeSuffix())); } else { ui->filenameLabel->setText(tr("Opened %1").arg(fileInfo.completeBaseName()+"."+fileInfo.completeSuffix())); // Open and import the file loadFile(); } } void QGCDataPlot2D::loadRawLog(QString file, QString xAxisName, QString yAxisFilter) { if (logFile != NULL) { logFile->close(); delete logFile; } // Postprocess log file logFile = new QTemporaryFile(); compressor = new LogCompressor(file, logFile->fileName()); compressor->startCompression(); // Block UI QProgressDialog progress("Transforming RAW log file to CSV", "Abort Transformation", 0, 1, this); progress.setWindowModality(Qt::WindowModal); while (!compressor->isFinished()) { MG::SLEEP::usleep(100000); progress.setMaximum(compressor->getDataLines()); progress.setValue(compressor->getCurrentLine()); } // Enforce end progress.setMaximum(compressor->getDataLines()); progress.setValue(compressor->getDataLines()); // Done with preprocessing - now load csv log loadCsvLog(logFile->fileName(), xAxisName, yAxisFilter); } /** * This function loads a CSV file into the plot. It tries to assign the dimension names * based on the first data row and tries to guess the separator char. * * @param file Name of the file to open * @param xAxisName Optional paramater. If given, the x axis dimension will be selected to match this string * @param yAxisFilter Optional parameter. If given, only data dimension names present in the filter string will be * plotted * * @code * * QString file = "/home/user/datalog.txt"; // With header: xyz * QString xAxis = "x"; * QString yAxis = "z"; * * // Plotted result will be x vs z with y ignored. * @endcode */ void QGCDataPlot2D::loadCsvLog(QString file, QString xAxisName, QString yAxisFilter) { if (logFile != NULL) { logFile->close(); delete logFile; curveNames.clear(); } logFile = new QFile(file); // Load CSV data if (!logFile->open(QIODevice::ReadOnly | QIODevice::Text)) return; // Set plot title if (ui->plotTitle->text() != "") plot->setTitle(ui->plotTitle->text()); if (ui->plotXAxisLabel->text() != "") plot->setAxisTitle(QwtPlot::xBottom, ui->plotXAxisLabel->text()); if (ui->plotYAxisLabel->text() != "") plot->setAxisTitle(QwtPlot::yLeft, ui->plotYAxisLabel->text()); // Extract header // Read in values // Find all keys QTextStream in(logFile); // First line is header QString header = in.readLine(); bool charRead = false; QString separator = ""; QList sepCandidates; sepCandidates << '\t'; sepCandidates << ','; sepCandidates << ';'; sepCandidates << ' '; sepCandidates << '~'; sepCandidates << '|'; // Iterate until separator is found // or full header is parsed for (int i = 0; i < header.length(); i++) { if (sepCandidates.contains(header.at(i))) { // Separator found if (charRead) { separator += header[i]; } } else { // Char found charRead = true; // If the separator is not empty, this char // has been read after a separator, so detection // is now complete if (separator != "") break; } } QString out = separator; out.replace("\t", ""); ui->filenameLabel->setText(file.split("/").last().split("\\").last()+" Separator: \""+out+"\""); //qDebug() << "READING CSV:" << header; // Clear plot plot->removeData(); QVector xValues; QMap* > yValues; curveNames.append(header.split(separator, QString::SkipEmptyParts)); QString curveName; // Clear UI elements ui->xAxis->clear(); ui->yAxis->clear(); ui->xRegressionComboBox->clear(); ui->yRegressionComboBox->clear(); ui->regressionOutput->clear(); int curveNameIndex = 0; //int xValueIndex = curveNames.indexOf(xAxisName); QString xAxisFilter; if (xAxisName == "") { xAxisFilter = curveNames.first(); } else { xAxisFilter = xAxisName; } foreach(curveName, curveNames) { // Add to plot x axis selection ui->xAxis->addItem(curveName); // Add to regression selection ui->xRegressionComboBox->addItem(curveName); ui->yRegressionComboBox->addItem(curveName); if (curveName != xAxisFilter) { if ((yAxisFilter == "") || yAxisFilter.contains(curveName)) { yValues.insert(curveName, new QVector()); // Add separator starting with second item if (curveNameIndex > 0 && curveNameIndex < curveNames.count()) { ui->yAxis->setText(ui->yAxis->text()+"|"); } ui->yAxis->setText(ui->yAxis->text()+curveName); curveNameIndex++; } } } // Select current axis in UI ui->xAxis->setCurrentIndex(curveNames.indexOf(xAxisFilter)); // Read data double x,y; while (!in.atEnd()) { QString line = in.readLine(); QStringList values = line.split(separator, QString::SkipEmptyParts); foreach(curveName, curveNames) { bool okx; if (curveName == xAxisFilter) { // X AXIS HANDLING // Take this value as x if it is selected x = values.at(curveNames.indexOf(curveName)).toDouble(&okx); xValues.append(x);// - 1270125570000ULL); } else { // Y AXIS HANDLING if(yAxisFilter == "" || yAxisFilter.contains(curveName)) { // Only append y values where a valid x value is present if (yValues.value(curveName)->count() == xValues.count() - 1) { bool oky; int curveNameIndex = curveNames.indexOf(curveName); if (values.count() > curveNameIndex) { y = values.at(curveNameIndex).toDouble(&oky); yValues.value(curveName)->append(y); } } } } } } // Add data array of each curve to the plot at once (fast) // Iterates through all x-y curve combinations for (int i = 0; i < yValues.count(); i++) { plot->appendData(yValues.keys().at(i), xValues.data(), yValues.values().at(i)->data(), xValues.count()); } plot->setStyleText(ui->style->currentText()); } bool QGCDataPlot2D::calculateRegression() { // TODO Add support for quadratic / cubic curve fitting return calculateRegression(ui->xRegressionComboBox->currentText(), ui->yRegressionComboBox->currentText(), "linear"); } /** * @param xName Name of the x dimension * @param yName Name of the y dimension * @param method Regression method, either "linear", "quadratic" or "cubic". Only linear is supported at this point */ bool QGCDataPlot2D::calculateRegression(QString xName, QString yName, QString method) { bool result = false; QString function; if (xName != yName) { if (QFileInfo(fileName).isReadable()) { loadCsvLog(fileName, xName, yName); ui->xRegressionComboBox->setCurrentIndex(curveNames.indexOf(xName)); ui->yRegressionComboBox->setCurrentIndex(curveNames.indexOf(yName)); } const int size = 100000; double x[size]; double y[size]; int copied = plot->data(yName, x, y, size); if (method == "linear") { double a; // Y-axis crossing double b; // Slope double r; // Regression coefficient int lin = linearRegression(x, y, copied, &a, &b, &r); if(lin == 1) { function = tr("%1 = %2 * %3 + %4 | R-coefficient: %5").arg(yName, QString::number(b), xName, QString::number(a), QString::number(r)); // Plot curve // y-axis crossing (x = 0) // Set plotting to lines only plot->appendData(tr("regression %1-%2").arg(xName, yName), 0.0, a); plot->setStyleText("lines"); // x-value of the current rightmost x position in the plot plot->appendData(tr("regression %1-%2").arg(xName, yName), plot->invTransform(QwtPlot::xBottom, plot->width() - plot->width()*0.08f), (a + b*plot->invTransform(QwtPlot::xBottom, plot->width() - plot->width() * 0.08f))); result = true; } else { function = tr("Linear regression failed. (Limit: %1 data points. Try with less)").arg(size); result = false; } } else if (method == "quadratic") { } else if (method == "cubic") { } else { function = tr("Regression method %1 not found").arg(method); result = false; } } else { // xName == yName function = tr("Please select different X and Y dimensions, not %1 = %2").arg(xName, yName); } ui->regressionOutput->setText(function); return result; } /** * Linear regression (least squares) for n data points. * Computes: * * y = a * x + b * * @param x values on x axis * @param y corresponding values on y axis * @param n Number of values * @param a returned slope of line * @param b y-axis intersection * @param r regression coefficient. The larger the coefficient is, the better is * the match of the regression. * @return 1 on success, 0 on failure (e.g. because of infinite slope) */ int QGCDataPlot2D::linearRegression(double* x,double* y,int n,double* a,double* b,double* r) { int i; double sumx=0,sumy=0,sumx2=0,sumy2=0,sumxy=0; double sxx,syy,sxy; *a = 0; *b = 0; *r = 0; if (n < 2) return(FALSE); /* Conpute some things we need */ for (i=0;icopy(fileName); qDebug() << "Saved CSV log. Success: " << success; //qDebug() << "READE TO SAVE CSV LOG TO " << fileName; } QGCDataPlot2D::~QGCDataPlot2D() { delete ui; } void QGCDataPlot2D::changeEvent(QEvent *e) { QWidget::changeEvent(e); switch (e->type()) { case QEvent::LanguageChange: ui->retranslateUi(this); break; default: break; } }