/*
*openPilotLog - A FOSS Pilot Logbook Application
*Copyright (C) 2020-2021 Felix Turowsky
*
*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, see .
*/
#include "firstrundialog.h"
#include "ui_firstrundialog.h"
#include "src/functions/alog.h"
#include "src/database/adatabase.h"
#include "src/gui/widgets/backupwidget.h"
//#include "src/database/adatabasesetup.h"
#include "src/database/adbsetup.h"
#include "src/classes/apilotentry.h"
#include "src/classes/adownload.h"
#include "src/classes/asettings.h"
#include "src/opl.h"
#include "src/functions/adate.h"
#include
#include
#include
#include "src/classes/astyle.h"
#include "src/functions/adatetime.h"
FirstRunDialog::FirstRunDialog(QWidget *parent) :
QDialog(parent),
ui(new Ui::FirstRunDialog)
{
ui->setupUi(this);
ui->stackedWidget->setCurrentIndex(0);
ui->lastnameLineEdit->setFocus();
ui->previousPushButton->setEnabled(false);
ui->logoLabel->setPixmap(QPixmap(Opl::Assets::LOGO));
// approach Combo Box
for (const auto &approach : Opl::ApproachTypes){
ui->approachComboBox->addItem(approach);
}
// Style combo box
const QSignalBlocker blocker_style(ui->styleComboBox);
ui->styleComboBox->addItems(AStyle::styles);
for (const auto &style_sheet : AStyle::styleSheets) {
ui->styleComboBox->addItem(style_sheet.styleSheetName);
}
ui->styleComboBox->addItem(QStringLiteral("Dark-Palette"));
ui->styleComboBox->model()->sort(0);
ui->styleComboBox->setCurrentText(AStyle::defaultStyle);
// Prepare Date Edits
dateEdits = this->findChildren();
for (const auto &date_format : ADate::getDisplayNames())
ui->dateFormatComboBox->addItem(date_format);
// Set Date Edits for currencies
for (const auto &date_edit : qAsConst(dateEdits)) {
date_edit->setDisplayFormat(
ADate::getFormatString(Opl::Date::ADateFormat::ISODate));
date_edit->setDate(QDate::currentDate());
}
// Debug - use ctrl + t to enable branchLineEdit to select from which git branch the templates are pulled
ui->branchLineEdit->setVisible(false);
}
FirstRunDialog::~FirstRunDialog()
{
delete ui;
}
void FirstRunDialog::on_previousPushButton_clicked()
{
auto current_index = ui->stackedWidget->currentIndex();
switch (current_index) {
case 0:
return;
case 1:
ui->previousPushButton->setEnabled(false);
break;
case 2:
ui->nextPushButton->setText(tr("Next"));
break;
}
ui->stackedWidget->setCurrentIndex(current_index - 1);
}
void FirstRunDialog::on_nextPushButton_clicked()
{
auto current_index = ui->stackedWidget->currentIndex();
switch (current_index) {
case 0:
if(ui->firstnameLineEdit->text().isEmpty()
|| ui->lastnameLineEdit->text().isEmpty())
{
QMessageBox(QMessageBox::Warning, tr("Error"),
tr("Please enter first and last name")
).exec();
return;
}
ui->previousPushButton->setEnabled(true);
break;
case 3:
ui->nextPushButton->setText(tr("Done"));
break;
case 4:
ui->nextPushButton->setDisabled(true);
if(!finishSetup())
QDialog::reject();
else
QDialog::accept();
return;
}
ui->stackedWidget->setCurrentIndex(current_index + 1);
}
bool FirstRunDialog::finishSetup()
{
writeSettings();
QFileInfo database_file(AStandardPaths::directory(AStandardPaths::Database).
absoluteFilePath(QStringLiteral("logbook.db")));
if (database_file.exists()) {
QMessageBox message_box(QMessageBox::Question, tr("Existing Database found"),
tr("An existing database file has been detected on your system.
"
"Would you like to create a backup of the existing database?
"
"Note: if you select no, the existing database will be overwritten. This"
"action is irreversible."),
QMessageBox::Yes | QMessageBox::No, this);
message_box.setDefaultButton(QMessageBox::Yes);
if(message_box.exec() == QMessageBox::Yes) {
// Create Backup
const QString backup_name = BackupWidget::absoluteBackupPath();
QFile old_db_file(database_file.absoluteFilePath());
if (!old_db_file.copy(backup_name)) {
WARN(tr("Unable to backup old database:
%1").arg(old_db_file.errorString()));
return false;
} else {
INFO(tr("Backup successfully created."));
}
}
//delete existing DB file
QFile db_file(database_file.absoluteFilePath());
if (!db_file.remove()) {
WARN(tr("Unable to delete existing database file."));
return false;
}
} // if database file exists
if (!aDB->connect()) {
QMessageBox message_box(QMessageBox::Critical, tr("Database setup failed"),
tr("Errors have ocurred creating the database."
"Without a working database The application will not be usable.
"
"The following error has ocurred:
"
"Database: Unable to connect"));
message_box.exec();
return false;
}
if (!setupDatabase()) {
QMessageBox message_box(QMessageBox::Critical, tr("Database setup failed"),
tr("Errors have ocurred creating the database."
"Without a working database The application will not be usable.
"
"The following error has ocurred:
%1"
).arg(aDB->lastError.text()));
message_box.exec();
return false;
}
if (!createUserEntry()) {
QMessageBox message_box(QMessageBox::Critical, tr("Database setup failed"),
tr("Unable to execute database query
"
"The following error has occured:
%1"
).arg(aDB->lastError.text()));
message_box.exec();
return false;
}
if (!writeCurrencies()) {
QMessageBox message_box(QMessageBox::Critical, tr("Database setup failed"),
tr("Unable to execute database query
"
"The following error has occured:
%1"
).arg(aDB->lastError.text()));
message_box.exec();
return false;
}
aDB->disconnect(); // connection will be re-established by main()
return true;
}
bool FirstRunDialog::downloadTemplates(QString branch_name)
{
// Create url string
auto template_url_string = QStringLiteral("https://raw.githubusercontent.com/fiffty-50/openpilotlog/");
template_url_string.append(branch_name);
template_url_string.append(QLatin1String("/assets/database/templates/"));
QDir template_dir(AStandardPaths::directory(AStandardPaths::Templates));
const auto template_tables = aDB->getTemplateTableNames();
for (const auto& table : template_tables) {
QEventLoop loop;
ADownload* dl = new ADownload;
QObject::connect(dl, &ADownload::done, &loop, &QEventLoop::quit );
dl->setTarget(QUrl(template_url_string + table + QLatin1String(".json")));
dl->setFileName(template_dir.absoluteFilePath(table + QLatin1String(".json")));
DEB << "Downloading: " << template_url_string + table + QLatin1String(".json");
DEB << "To:" << AStandardPaths::directory(AStandardPaths::Templates);
dl->download();
dl->deleteLater();
loop.exec(); // event loop waits for download done signal before allowing loop to continue
QFileInfo downloaded_file(template_dir.filePath(table + QLatin1String(".json")));
if (downloaded_file.size() == 0)
return false; // ssl/network error
}
return true;
}
void FirstRunDialog::writeSettings()
{
ASettings::resetToDefaults();
ASettings::write(ASettings::FlightLogging::Function, ui->functionComboBox->currentText());
ASettings::write(ASettings::FlightLogging::Approach, ui->approachComboBox->currentIndex());
switch (ui->nightComboBox->currentIndex()) {
case 0:
ASettings::write(ASettings::FlightLogging::NightLoggingEnabled, true);
break;
case 1:
ASettings::write(ASettings::FlightLogging::NightLoggingEnabled, false);
break;
default:
ASettings::write(ASettings::FlightLogging::NightLoggingEnabled, true);
break;
}
switch (ui->nightRulesComboBox->currentIndex()) {
case 0:
ASettings::write(ASettings::FlightLogging::NightAngle, 6);
break;
case 1:
ASettings::write(ASettings::FlightLogging::NightAngle, 0);
break;
}
ASettings::write(ASettings::FlightLogging::LogIFR, ui->rulesComboBox->currentIndex());
ASettings::write(ASettings::FlightLogging::FlightNumberPrefix, ui->prefixLineEdit->text());
ASettings::write(ASettings::FlightLogging::FlightTimeFormat, Opl::Time::Default);
ASettings::write(ASettings::UserData::DisplaySelfAs, ui->aliasComboBox->currentIndex());
ASettings::write(ASettings::Main::LogbookView, ui->logbookViewComboBox->currentIndex());
switch (ui->currWarningCheckBox->checkState()) {
case Qt::CheckState::Checked:
ASettings::write(ASettings::UserData::CurrWarningEnabled, true);
break;
case Qt::CheckState::Unchecked:
ASettings::write(ASettings::UserData::CurrWarningEnabled, false);
break;
default:
break;
}
ASettings::write(ASettings::UserData::CurrWarningThreshold, ui->currWarningThresholdSpinBox->value());
ASettings::write(ASettings::Main::Style, ui->styleComboBox->currentText());
QSettings settings;
settings.sync();
}
bool FirstRunDialog::setupDatabase()
{
QMessageBox confirm(QMessageBox::Question, tr("Create Database"),
tr("We are now going to create the database.
"
"Would you like to download the latest database information?"
"
(Recommended, Internet connection required)"),
QMessageBox::Yes | QMessageBox::No, this);
confirm.setDefaultButton(QMessageBox::No);
if (confirm.exec() == QMessageBox::Yes) {
useRessourceData = false;
if (!downloadTemplates(ui->branchLineEdit->text())) {
QMessageBox message_box(this);
message_box.setText(tr("Downloading latest data has failed.
Using local data instead."));
message_box.exec();
useRessourceData = true; // fall back
}
} else {
useRessourceData = true;
}
if(!aDbSetup::createDatabase()) {
WARN(tr("Database creation has been unsuccessful. The following error has ocurred:
%1")
.arg(aDB->lastError.text()));
return false;
}
if(!aDbSetup::importTemplateData(useRessourceData)) {
WARN(tr("Database creation has been unsuccessful. Unable to fill template data.
%1")
.arg(aDB->lastError.text()));
return false;
}
return true;
}
bool FirstRunDialog::createUserEntry()
{
QMap data;
data.insert(Opl::Db::PILOTS_LASTNAME, ui->lastnameLineEdit->text());
data.insert(Opl::Db::PILOTS_FIRSTNAME, ui->firstnameLineEdit->text());
data.insert(Opl::Db::PILOTS_ALIAS, QStringLiteral("self"));
data.insert(Opl::Db::PILOTS_EMPLOYEEID, ui->employeeidLineEdit->text());
data.insert(Opl::Db::PILOTS_PHONE, ui->phoneLineEdit->text());
data.insert(Opl::Db::PILOTS_EMAIL, ui->emailLineEdit->text());
auto pilot = APilotEntry(1);
pilot.setData(data);
return aDB->commit(pilot);
}
bool FirstRunDialog::writeCurrencies()
{
const QList> currencies_list = {
{ACurrencyEntry::CurrencyName::Licence, ui->currLicDateEdit},
{ACurrencyEntry::CurrencyName::TypeRating, ui->currTrDateEdit},
{ACurrencyEntry::CurrencyName::LineCheck, ui->currLckDateEdit},
{ACurrencyEntry::CurrencyName::Medical, ui->currMedDateEdit},
{ACurrencyEntry::CurrencyName::Custom1, ui->currCustom1DateEdit},
{ACurrencyEntry::CurrencyName::Custom2, ui->currCustom1DateEdit},
};
QDate today = QDate::currentDate();
for (const auto &pair : currencies_list) {
// only write dates that have been edited
if (pair.second->date() != today) {
ACurrencyEntry entry(pair.first, pair.second->date());
if (!aDB->commit(entry))
return false;
}
}
return true;
}
void FirstRunDialog::reject()
{
QMessageBox confirm(QMessageBox::Critical,
tr("Setup incomplete"),
tr("Without completing the initial setup"
" you cannot use the application.
"
"Quit anyway?"),
QMessageBox::Yes | QMessageBox::No, this);
confirm.setDefaultButton(QMessageBox::No);
if (confirm.exec() == QMessageBox::Yes) {
DEB << "rejected.";
QDialog::reject();
}
}
void FirstRunDialog::keyPressEvent(QKeyEvent *keyEvent)
{
if(keyEvent->type() == QKeyEvent::KeyPress) {
if(keyEvent->matches(QKeySequence::AddTab)) {
ui->branchLineEdit->setVisible(true);
ui->branchLineEdit->setEnabled(true);
}
}
}
void FirstRunDialog::on_styleComboBox_currentTextChanged(const QString &new_style_setting)
{
DEB << "style selected:"<currCustom1LineEdit->text());
}
void FirstRunDialog::on_currCustom2LineEdit_editingFinished()
{
ASettings::write(ASettings::UserData::Custom2CurrencyName, ui->currCustom2LineEdit->text());
}
void FirstRunDialog::on_dateFormatComboBox_currentIndexChanged(int index)
{
Opl::Date::ADateFormat format = (Opl::Date::ADateFormat)index;
for (const auto &date_edit : qAsConst(dateEdits)) {
date_edit->setDisplayFormat(
ADate::getFormatString(format));
}
ASettings::write(ASettings::Main::DateFormat, index);
}
void FirstRunDialog::on_importPushButton_clicked()
{
QString filename = QFileDialog::getOpenFileName(
this,
tr("Choose backup file"),
QDir::homePath(),
"*.db"
);
if(filename.isEmpty()) { // QFileDialog has been cancelled
WARN(tr("No Database has been selected."));
return;
}
QMessageBox confirm(this);
confirm.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
confirm.setDefaultButton(QMessageBox::No);
confirm.setIcon(QMessageBox::Question);
confirm.setWindowTitle(tr("Import Database"));
confirm.setText(tr("The following database will be imported:
"
"%1
"
"
Is this correct?"
).arg(aDB->databaseSummaryString(filename)));
if (confirm.exec() == QMessageBox::Yes) {
if(!aDB->restoreBackup(filename)) {
WARN(tr("Unable to import database file:").arg(filename));
return;
}
INFO(tr("Database successfully imported."));
QDialog::accept(); // quit the dialog as if a database was successfully created
} else {
return;
}
}