/* *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; } }