Files
GoldenCheetah/src/GcCrashDialog.cpp
Mark Liversedge aa8605e8d5 QT5 -- 1 of 3
Porting the codebase to QT 5 (5.2) to get the
latest bug fixes, performance and improved platform
support.

This first part is to fixup the codebase to compile
on Qt 5, but some aspects have been broken (video).

The second part is to migrate from Qwt 6.0.1 to the
latest Qwt for multiaxis support.

The third part will be to fixup any platform specific
issues or issues identified at runtime.
2013-12-09 09:57:13 +00:00

366 lines
11 KiB
C++

/*
* Copyright (c) 2013 Mark Liversedge (liversedge@gmail.com)
*
* 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 2 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., 51
* Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "GcCrashDialog.h"
#include "Settings.h"
#include "GcUpgrade.h"
#include <QtGui>
#include <QWebFrame>
#include "DBAccess.h"
#include "MetricAggregator.h"
#include <QtSql>
#include <qwt_plot_curve.h>
#define GCC_VERSION QString("%1.%2.%3").arg(__GNUC__).arg(__GNUC_MINOR__).arg(__GNUC_PATCHLEVEL__)
#ifdef GC_HAVE_QWTPLOT3D
#include "qwt3d_global.h"
#endif
#ifdef GC_HAVE_ICAL
#include "ICalendar.h"
#endif
#ifdef GC_HAVE_D2XX
#include "D2XX.h"
#endif
#ifdef GC_HAVE_SRMIO
#include "srmio.h"
#endif
#ifdef GC_HAVE_LIBOAUTH
#include <oauth.h>
#endif
#ifdef GC_HAVE_LUCENE
#include "Lucene.h"
#endif
#ifdef GC_HAVE_WFAPI
#include "WFApi.h"
#endif
GcCrashDialog::GcCrashDialog(QDir home) : QDialog(NULL, Qt::Dialog), home(home)
{
setAttribute(Qt::WA_DeleteOnClose, true); // caller must delete me, once they've extracted the name
setWindowTitle(QString(tr("%1 Crash Recovery").arg(home.dirName())));
QVBoxLayout *layout = new QVBoxLayout(this);
QHBoxLayout *toprow = new QHBoxLayout;
QPushButton *critical = new QPushButton(style()->standardIcon(QStyle::SP_MessageBoxCritical), "", this);
critical->setFixedSize(128,128);
critical->setFlat(true);
critical->setIconSize(QSize(120,120));
critical->setAutoFillBackground(false);
critical->setFocusPolicy(Qt::NoFocus);
QLabel *header = new QLabel(this);
header->setWordWrap(true);
header->setTextFormat(Qt::RichText);
header->setText(tr("<b>GoldenCheetah appears to have PREVIOUSLY crashed for this athlete. "
"</b><br><br>The report below gives some diagnostic information "
"that will be useful to the developers. Feel free to post this with a "
"short description of what was occurring when the crash happened to the "
"<href a=\"https://groups.google.com/forum/?fromgroups=#!forum/golden-cheetah-users\">"
"GoldenCheetah forums</href><br>"
"<b><br>We respect privacy - this log does NOT contain ids, passwords or personal information.</b><br>"
"<b><br>When this dialog is closed the athlete will be opened.</b>"));
toprow->addWidget(critical);
toprow->addWidget(header);
layout->addLayout(toprow);
report = new QWebView(this);
report->setContentsMargins(0,0,0,0);
report->page()->view()->setContentsMargins(0,0,0,0);
report->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
report->setAcceptDrops(false);
QFont defaultFont; // mainwindow sets up the defaults.. we need to apply
report->settings()->setFontSize(QWebSettings::DefaultFontSize, defaultFont.pointSize()+1);
report->settings()->setFontFamily(QWebSettings::StandardFont, defaultFont.family());
layout->addWidget(report);
QHBoxLayout *lastRow = new QHBoxLayout;
QPushButton *saveAsButton = new QPushButton(tr("Save Crash Report..."), this);
connect(saveAsButton, SIGNAL(clicked()), this, SLOT(saveAs()));
QPushButton *OKButton = new QPushButton(tr("Close"), this);
connect(OKButton, SIGNAL(clicked()), this, SLOT(accept()));
lastRow->addWidget(saveAsButton);
lastRow->addStretch();
lastRow->addWidget(OKButton);
layout->addLayout(lastRow);
setHTML();
}
QString GcCrashDialog::versionHTML()
{
// -- OS ----
QString os = "";
#ifdef Q_OS_LINUX
os = "Linux";
#endif
#ifdef WIN32
os = "Win";
#endif
#ifdef Q_OS_MAC
os = QString("Mac OS X 10.%1").arg(QSysInfo::MacintoshVersion - 2);
if (QSysInfo::MacintoshVersion == QSysInfo::MV_SNOWLEOPARD)
os += " Snow Leopard";
else if (QSysInfo::MacintoshVersion == QSysInfo::MV_LION)
os += " Lion";
else if (QSysInfo::MacintoshVersion == 10)
os += " Mountain Lion";
#endif
// -- SCHEMA VERSION ----
QString schemaVersion = QString("%1").arg(DBSchemaVersion);
// -- SRMIO ----
QString srmio = "none";
#ifdef GC_HAVE_SRMIO
#ifdef SRMIO_VERSION
srmio = QString("%1 %2").arg(SRMIO_VERSION).arg(srmio_commit);
#else
srmio = "yes";
#endif
#endif
// -- D2XX ----
QString d2xx = "none";
#ifdef GC_HAVE_D2XX
d2xx = "yes";
#endif
// -- LIBOAUTH ----
QString oauth = "none";
#ifdef GC_HAVE_LIBOAUTH
oauth = LIBOAUTH_VERSION;
#endif
// -- QWTPLOT3D ----
QString qwtplot3d = "none";
#ifdef GC_HAVE_QWTPLOT3D
qwtplot3d = QString::number(QWT3D_MAJOR_VERSION) + "."
+ QString::number(QWT3D_MINOR_VERSION) + "."
+ QString::number(QWT3D_PATCH_VERSION);
#endif
// -- KML ----
QString kml = "none";
#ifdef GC_HAVE_KML
kml = "yes";
#endif
// -- ICAL ----
QString ical = "none";
#ifdef GC_HAVE_ICAL
ical = ICAL_VERSION;
#endif
// -- USBXPRESS ----
QString usbxpress = "none";
#ifdef GC_HAVE_USBXPRESS
usbxpress = "yes";
#endif
// -- LIBUSB ----
QString libusb = "none";
#ifdef GC_HAVE_LIBUSB
libusb = "yes";
#endif
// -- VLC ----
QString vlc = "none";
#ifdef GC_HAVE_VLC
vlc = "yes";
#endif
// -- LUCENE ----
QString clucene = "none";
#ifdef GC_HAVE_LUCENE
clucene = _CL_VERSION;
#endif
#ifdef GC_HAVE_WFAPI
QString wfapi = WFApi::getInstance()->apiVersion();
#else
QString wfapi = QString("none");
#endif
QString gc_version = tr(
"<p>Build date: %1 %2"
"<br>Build id: %3"
"<br>Version: %4"
"<br>DB Schema: %5"
"<br>OS: %6"
"<br>")
.arg(__DATE__)
.arg(__TIME__)
.arg(GcUpgrade::version())
#ifdef GC_VERSION
.arg(GC_VERSION)
#else
.arg("(developer build)")
#endif
.arg(schemaVersion)
.arg(os);
QString lib_version = tr(
"<table>"
"<tr><td colspan=\"2\">QT</td><td>%1</td></tr>"
"<tr><td colspan=\"2\">QWT</td><td>%2</td></tr>"
"<tr><td colspan=\"2\">GCC</td><td>%3</td></tr>"
"<tr><td colspan=\"2\">SRMIO</td><td>%4</td></tr>"
"<tr><td colspan=\"2\">OAUTH</td><td>%5</td></tr>"
"<tr><td colspan=\"2\">D2XX</td><td>%6</td></tr>"
"<tr><td colspan=\"2\">QWTPLOT3D</td><td>%7</td></tr>"
"<tr><td colspan=\"2\">KML</td><td>%8</td></tr>"
"<tr><td colspan=\"2\">ICAL</td><td>%9</td></tr>"
"<tr><td colspan=\"2\">USBXPRESS</td><td>%10</td></tr>"
"<tr><td colspan=\"2\">LIBUSB</td><td>%11</td></tr>"
"<tr><td colspan=\"2\">Wahoo API</td><td>%12</td></tr>"
"<tr><td colspan=\"2\">VLC</td><td>%13</td></tr>"
"<tr><td colspan=\"2\">LUCENE</td><td>%14</td></tr>"
"</table>"
)
.arg(QT_VERSION_STR)
.arg(QWT_VERSION_STR)
.arg(GCC_VERSION)
.arg(srmio)
.arg(oauth)
.arg(d2xx)
.arg(qwtplot3d)
.arg(kml)
.arg(ical)
.arg(usbxpress)
.arg(libusb)
.arg(wfapi)
.arg(vlc)
.arg(clucene)
;
QString versionText = QString("<center>" + gc_version + lib_version + "</center>");
return versionText;
}
void
GcCrashDialog::setHTML()
{
QString text;
// the cyclist...
text += QString("<center><h3>Cyclist: \"%1\"</h3></center><br>").arg(home.dirName());
// version info
text += "<center><h3>Version Info</h3></center>";
text += versionHTML();
// metric log...
text += "<center><h3>Metric Log</h3></center>";
text += "<center><table border=0 cellspacing=10 width=\"90%\">";
QFile metriclog(home.absolutePath() + "/" + "metric.log");
if (metriclog.open(QIODevice::ReadOnly)) {
// read in line by line and add to diag file
QTextStream in(&metriclog);
while (!in.atEnd()) {
QString lines = in.readLine();
foreach(QString line, lines.split('\r')) {
text += "<tr><td align=\"center\">" + line + "</td></tr>";
}
}
metriclog.close();
}
text += "</table></center>";
// files...
text += "<center><h3>Athlete Directory</h3></center>";
text += "<center><table border=0 cellspacing=10 width=\"90%\">";
foreach(QString file, home.entryList(QDir::NoFilter, QDir::Time)) {
text += QString("<tr><td align=\"right\"> %1</td><td align=\"left\">%2</td></tr>")
.arg(file)
.arg(QFileInfo(home.absolutePath() + "/" + file).lastModified().toString());
}
text += "</table></center>";
// settings...
text += "<center><h3>All Settings</h3></center>";
text += "<center><table border=0 cellspacing=10 width=\"90%\">";
foreach(QString key, appsettings->allKeys()) {
// RESPECT PRIVACY
// we do not disclose user names and passwords or key ids
if (key.endsWith("/user") || key.endsWith("/pass") || key.endsWith("/key")) continue;
// we do not disclose personally identifiable information
if (key.endsWith("/dob") || key.endsWith("/weight") ||
key.endsWith("/sex") || key.endsWith("/bio")) continue;
text += QString("<tr><td align=\"right\" width=\"50%\"> %1</td><td align=\"left\">"
"<span style=\"max-width:50%;\">%2</span></td></tr>")
.arg(key)
.arg(appsettings->value(NULL, key).toString().leftJustified(60, ' ', true));
}
text += "</table></center>";
report->page()->mainFrame()->setHtml(text);
}
void
GcCrashDialog::saveAs()
{
QString fileName = QFileDialog::getSaveFileName( this, tr("Save Diagnostics"), QDir::homePath(), tr("Text File (*.txt)"));
// now write to it
QFile file(fileName);
file.resize(0);
QTextStream out(&file);
out.setCodec("UTF-8");
if (file.open(QIODevice::WriteOnly)) {
// write the texts
out << report->page()->mainFrame()->toPlainText();
}
file.close();
}