mirror of
https://github.com/subsurface/subsurface.git
synced 2024-11-28 21:20:19 +00:00
315 lines
12 KiB
C++
315 lines
12 KiB
C++
#include <QFileDialog>
|
|
#include <QShortcut>
|
|
#include <QSettings>
|
|
#include <QtConcurrent>
|
|
|
|
#include "desktop-widgets/divelogexportdialog.h"
|
|
#include "core/divelogexportlogic.h"
|
|
#include "desktop-widgets/diveshareexportdialog.h"
|
|
#include "ui_divelogexportdialog.h"
|
|
#include "desktop-widgets/subsurfacewebservices.h"
|
|
#include "core/worldmap-save.h"
|
|
#include "core/save-html.h"
|
|
#include "desktop-widgets/mainwindow.h"
|
|
#include "profile-widget/profilewidget2.h"
|
|
|
|
|
|
#define GET_UNIT(name, field, f, t) \
|
|
v = settings.value(QString(name)); \
|
|
if (v.isValid()) \
|
|
field = (v.toInt() == 0) ? (t) : (f); \
|
|
else \
|
|
field = default_prefs.units.field
|
|
|
|
DiveLogExportDialog::DiveLogExportDialog(QWidget *parent) : QDialog(parent),
|
|
ui(new Ui::DiveLogExportDialog)
|
|
{
|
|
ui->setupUi(this);
|
|
showExplanation();
|
|
QShortcut *quit = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_Q), this);
|
|
connect(quit, SIGNAL(activated()), MainWindow::instance(), SLOT(close()));
|
|
QShortcut *close = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_W), this);
|
|
connect(close, SIGNAL(activated()), this, SLOT(close()));
|
|
|
|
/* the names are not the actual values exported to the json files,The font-family property should hold several
|
|
font names as a "fallback" system, to ensure maximum compatibility between browsers/operating systems */
|
|
ui->fontSelection->addItem("Arial", "Arial, Helvetica, sans-serif");
|
|
ui->fontSelection->addItem("Impact", "Impact, Charcoal, sans-serif");
|
|
ui->fontSelection->addItem("Georgia", "Georgia, serif");
|
|
ui->fontSelection->addItem("Courier", "Courier, monospace");
|
|
ui->fontSelection->addItem("Verdana", "Verdana, Geneva, sans-serif");
|
|
|
|
QSettings settings;
|
|
settings.beginGroup("HTML");
|
|
if (settings.contains("fontSelection")) {
|
|
ui->fontSelection->setCurrentIndex(settings.value("fontSelection").toInt());
|
|
}
|
|
if (settings.contains("fontSizeSelection")) {
|
|
ui->fontSizeSelection->setCurrentIndex(settings.value("fontSizeSelection").toInt());
|
|
}
|
|
if (settings.contains("themeSelection")) {
|
|
ui->themeSelection->setCurrentIndex(settings.value("themeSelection").toInt());
|
|
}
|
|
if (settings.contains("subsurfaceNumbers")) {
|
|
ui->exportSubsurfaceNumber->setChecked(settings.value("subsurfaceNumbers").toBool());
|
|
}
|
|
if (settings.contains("yearlyStatistics")) {
|
|
ui->exportStatistics->setChecked(settings.value("yearlyStatistics").toBool());
|
|
}
|
|
if (settings.contains("listOnly")) {
|
|
ui->exportListOnly->setChecked(settings.value("listOnly").toBool());
|
|
}
|
|
if (settings.contains("exportPhotos")) {
|
|
ui->exportPhotos->setChecked(settings.value("exportPhotos").toBool());
|
|
}
|
|
settings.endGroup();
|
|
}
|
|
|
|
DiveLogExportDialog::~DiveLogExportDialog()
|
|
{
|
|
delete ui;
|
|
}
|
|
|
|
void DiveLogExportDialog::showExplanation()
|
|
{
|
|
if (ui->exportUDDF->isChecked()) {
|
|
ui->description->setText(tr("Generic format that is used for data exchange between a variety of diving related programs."));
|
|
} else if (ui->exportCSV->isChecked()) {
|
|
ui->description->setText(tr("Comma separated values describing the dive profile."));
|
|
} else if (ui->exportCSVDetails->isChecked()) {
|
|
ui->description->setText(tr("Comma separated values of the dive information. This includes most of the dive details but no profile information."));
|
|
} else if (ui->exportDivelogs->isChecked()) {
|
|
ui->description->setText(tr("Send the dive data to divelogs.de website."));
|
|
} else if (ui->exportDiveshare->isChecked()) {
|
|
ui->description->setText(tr("Send the dive data to dive-share.appspot.com website."));
|
|
} else if (ui->exportWorldMap->isChecked()) {
|
|
ui->description->setText(tr("HTML export of the dive locations, visualized on a world map."));
|
|
} else if (ui->exportSubsurfaceXML->isChecked()) {
|
|
ui->description->setText(tr("Subsurface native XML format."));
|
|
} else if (ui->exportImageDepths->isChecked()) {
|
|
ui->description->setText(tr("Write depths of images to file."));
|
|
} else if (ui->exportTeX->isChecked()) {
|
|
ui->description->setText(tr("Write dive as TeX macros to file."));
|
|
}
|
|
}
|
|
|
|
void DiveLogExportDialog::exportHtmlInit(const QString &filename)
|
|
{
|
|
struct htmlExportSetting hes;
|
|
hes.themeFile = (ui->themeSelection->currentText() == tr("Light")) ? "light.css" : "sand.css";
|
|
hes.exportPhotos = ui->exportPhotos->isChecked();
|
|
hes.selectedOnly = ui->exportSelectedDives->isChecked();
|
|
hes.listOnly = ui->exportListOnly->isChecked();
|
|
hes.fontFamily = ui->fontSelection->itemData(ui->fontSelection->currentIndex()).toString();
|
|
hes.fontSize = ui->fontSizeSelection->currentText();
|
|
hes.themeSelection = ui->themeSelection->currentIndex();
|
|
hes.subsurfaceNumbers = ui->exportSubsurfaceNumber->isChecked();
|
|
hes.yearlyStatistics = ui->exportStatistics->isChecked();
|
|
|
|
exportHtmlInitLogic(filename, hes);
|
|
}
|
|
|
|
void DiveLogExportDialog::on_exportGroup_buttonClicked(QAbstractButton *button)
|
|
{
|
|
Q_UNUSED(button)
|
|
showExplanation();
|
|
}
|
|
|
|
void DiveLogExportDialog::on_buttonBox_accepted()
|
|
{
|
|
QString filename;
|
|
QString stylesheet;
|
|
QSettings settings;
|
|
QString lastDir = QDir::homePath();
|
|
|
|
settings.beginGroup("FileDialog");
|
|
if (settings.contains("LastDir")) {
|
|
if (QDir::setCurrent(settings.value("LastDir").toString())) {
|
|
lastDir = settings.value("LastDir").toString();
|
|
}
|
|
}
|
|
settings.endGroup();
|
|
|
|
switch (ui->tabWidget->currentIndex()) {
|
|
case 0:
|
|
if (ui->exportUDDF->isChecked()) {
|
|
stylesheet = "uddf-export.xslt";
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export UDDF file as"), lastDir,
|
|
tr("UDDF files (*.uddf *.UDDF)"));
|
|
} else if (ui->exportCSV->isChecked()) {
|
|
stylesheet = "xml2csv.xslt";
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export CSV file as"), lastDir,
|
|
tr("CSV files (*.csv *.CSV)"));
|
|
} else if (ui->exportCSVDetails->isChecked()) {
|
|
stylesheet = "xml2manualcsv.xslt";
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export CSV file as"), lastDir,
|
|
tr("CSV files (*.csv *.CSV)"));
|
|
} else if (ui->exportDivelogs->isChecked()) {
|
|
DivelogsDeWebServices::instance()->prepareDivesForUpload(ui->exportSelected->isChecked());
|
|
} else if (ui->exportDiveshare->isChecked()) {
|
|
DiveShareExportDialog::instance()->prepareDivesForUpload(ui->exportSelected->isChecked());
|
|
} else if (ui->exportWorldMap->isChecked()) {
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export world map"), lastDir,
|
|
tr("HTML files (*.html)"));
|
|
if (!filename.isNull() && !filename.isEmpty())
|
|
export_worldmap_HTML(filename.toUtf8().data(), ui->exportSelected->isChecked());
|
|
} else if (ui->exportSubsurfaceXML->isChecked()) {
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export Subsurface XML"), lastDir,
|
|
tr("XML files (*.xml *.ssrf)"));
|
|
if (!filename.isNull() && !filename.isEmpty()) {
|
|
if (!filename.contains('.'))
|
|
filename.append(".ssrf");
|
|
QByteArray bt = QFile::encodeName(filename);
|
|
save_dives_logic(bt.data(), ui->exportSelected->isChecked());
|
|
}
|
|
} else if (ui->exportImageDepths->isChecked()) {
|
|
filename = QFileDialog::getSaveFileName(this, tr("Save image depths"), lastDir);
|
|
if (!filename.isNull() && !filename.isEmpty())
|
|
export_depths(filename.toUtf8().data(), ui->exportSelected->isChecked());
|
|
} else if (ui->exportTeX->isChecked()) {
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export to TeX file"), lastDir, tr("TeX files (*.tex)"));
|
|
if (!filename.isNull() && !filename.isEmpty())
|
|
export_TeX(filename.toUtf8().data(), ui->exportSelected->isChecked());
|
|
}
|
|
break;
|
|
case 1:
|
|
filename = QFileDialog::getSaveFileName(this, tr("Export HTML files as"), lastDir,
|
|
tr("HTML files (*.html)"));
|
|
if (!filename.isNull() && !filename.isEmpty())
|
|
exportHtmlInit(filename);
|
|
break;
|
|
}
|
|
|
|
if (!filename.isNull() && !filename.isEmpty()) {
|
|
// remember the last export path
|
|
QFileInfo fileInfo(filename);
|
|
settings.beginGroup("FileDialog");
|
|
settings.setValue("LastDir", fileInfo.dir().path());
|
|
settings.endGroup();
|
|
// the non XSLT exports are called directly above, the XSLT based ons are called here
|
|
if (!stylesheet.isEmpty()) {
|
|
future = QtConcurrent::run(export_dives_xslt, filename.toUtf8(), ui->exportSelected->isChecked(), ui->CSVUnits_2->currentIndex(), stylesheet.toUtf8());
|
|
MainWindow::instance()->getNotificationWidget()->showNotification(tr("Please wait, exporting..."), KMessageWidget::Information);
|
|
MainWindow::instance()->getNotificationWidget()->setFuture(future);
|
|
}
|
|
}
|
|
}
|
|
|
|
void DiveLogExportDialog::export_depths(const char *filename, const bool selected_only)
|
|
{
|
|
FILE *f;
|
|
struct dive *dive;
|
|
depth_t depth;
|
|
int i;
|
|
const char *unit = NULL;
|
|
|
|
struct membuffer buf = {};
|
|
|
|
for_each_dive (i, dive) {
|
|
if (selected_only && !dive->selected)
|
|
continue;
|
|
|
|
FOR_EACH_PICTURE (dive) {
|
|
int n = dive->dc.samples;
|
|
struct sample *s = dive->dc.sample;
|
|
depth.mm = 0;
|
|
while (--n >= 0 && (int32_t)s->time.seconds <= picture->offset.seconds) {
|
|
depth.mm = s->depth.mm;
|
|
s++;
|
|
}
|
|
put_format(&buf, "%s\t%.1f", picture->filename, get_depth_units(depth.mm, NULL, &unit));
|
|
put_format(&buf, "%s\n", unit);
|
|
}
|
|
}
|
|
|
|
f = subsurface_fopen(filename, "w+");
|
|
if (!f) {
|
|
report_error(tr("Can't open file %s").toUtf8().data(), filename);
|
|
} else {
|
|
flush_buffer(&buf, f); /*check for writing errors? */
|
|
fclose(f);
|
|
}
|
|
free_buffer(&buf);
|
|
}
|
|
|
|
void DiveLogExportDialog::export_TeX(const char *filename, const bool selected_only)
|
|
{
|
|
FILE *f;
|
|
struct dive *dive;
|
|
int i;
|
|
bool need_pagebreak = false;
|
|
|
|
struct membuffer buf = {};
|
|
|
|
put_format(&buf, "\\input subsurfacetemplate\n");
|
|
put_format(&buf, "%% This is a plain TeX file. Compile with pdftex, not pdflatex!\n");
|
|
put_format(&buf, "%% You will also need a subsurfacetemplate.tex in the current directory.\n");
|
|
put_format(&buf, "%% You can downlaod an example from http://www.atdotde.de/~robert/subsurfacetemplate\n%%\n");
|
|
for_each_dive (i, dive) {
|
|
if (selected_only && !dive->selected)
|
|
continue;
|
|
|
|
QString filename = "profile%1.png";
|
|
ProfileWidget2 *profile = MainWindow::instance()->graphics();
|
|
profile->plotDive(dive, true);
|
|
profile->setToolTipVisibile(false);
|
|
QPixmap pix = QPixmap::grabWidget(profile);
|
|
profile->setToolTipVisibile(true);
|
|
pix.save(filename.arg(dive->number));
|
|
|
|
|
|
|
|
struct tm tm;
|
|
utc_mkdate(dive->when, &tm);
|
|
|
|
dive_site *site = get_dive_site_by_uuid(dive->dive_site_uuid);
|
|
QRegExp ct("countrytag: (\\w+)");
|
|
QString country;
|
|
if (ct.indexIn(site->notes) >= 0)
|
|
country = ct.cap(1);
|
|
else
|
|
country = "";
|
|
|
|
pressure_t delta_p = {.mbar = 0};
|
|
|
|
QString star = "*";
|
|
QString viz = star.repeated(dive->visibility);
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_CYLINDERS; i++)
|
|
if (is_cylinder_used(dive, i))
|
|
delta_p.mbar += dive->cylinder[i].start.mbar - dive->cylinder[i].end.mbar;
|
|
|
|
if (need_pagebreak)
|
|
put_format(&buf, "\\vfill\\eject\n");
|
|
need_pagebreak = true;
|
|
put_format(&buf, "\\def\\date{%04u-%02u-%02u}\n",
|
|
tm.tm_year, tm.tm_mon+1, tm.tm_mday);
|
|
put_format(&buf, "\\def\\number{%d}\n", dive->number);
|
|
put_format(&buf, "\\def\\place{%s}\n", site ? site->name : "");
|
|
put_format(&buf, "\\def\\spot{}\n");
|
|
put_format(&buf, "\\def\\country{%s}\n", country.toUtf8().data());
|
|
put_format(&buf, "\\def\\entrance{}\n");
|
|
put_format(&buf, "\\def\\time{%u:%02u}\n", FRACTION(dive->duration.seconds, 60));
|
|
put_format(&buf, "\\def\\depth{%u.%01um}\n", FRACTION(dive->maxdepth.mm / 100, 10));
|
|
put_format(&buf, "\\def\\gasuse{%u.%01ubar}\n", FRACTION(delta_p.mbar / 100, 10));
|
|
put_format(&buf, "\\def\\sac{%u.%01u l/min}\n", FRACTION(dive->sac/100,10));
|
|
put_format(&buf, "\\def\\type{%s}\n", dive->tag_list ? dive->tag_list->tag->name : "");
|
|
put_format(&buf, "\\def\\viz{%s}\n", viz.toUtf8().data());
|
|
put_format(&buf, "\\def\\plot{\\includegraphics[width=9cm,height=4cm]{profile%d}}\n", dive->number);
|
|
put_format(&buf, "\\def\\comment{%s}\n", dive->notes ? dive->notes : "");
|
|
put_format(&buf, "\\def\\buddy{%s}\n", dive->buddy ? dive->buddy : "");
|
|
put_format(&buf, "\\page\n");
|
|
}
|
|
|
|
put_format(&buf, "\\bye\n");
|
|
|
|
f = subsurface_fopen(filename, "w+");
|
|
if (!f) {
|
|
report_error(tr("Can't open file %s").toUtf8().data(), filename);
|
|
} else {
|
|
flush_buffer(&buf, f); /*check for writing errors? */
|
|
fclose(f);
|
|
}
|
|
free_buffer(&buf);
|
|
}
|