Newer
Older
/***************************************************************************
* Copyright (C) 2011-2020 by Tomasz Bojczuk *
* *
* 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 <http://www.gnu.org/licenses/>. *
***************************************************************************/
SeeLook
committed
#include <tmtr.h>
#include <tpath.h>
#include <tnootkaqml.h>
#include <tsound.h>
#include "main/tnameitem.h"
#include "main/tmainscoreobject.h"
#include "dialogs/tdialogloaderobject.h"
#include "help/tmainhelp.h"
SeeLook
committed
#if defined (Q_OS_ANDROID)
#include <Android/tandroid.h>
#include <QtWidgets/qapplication.h>
#include <QtGui/qicon.h>
#include <QtGui/qpalette.h>
#include <QtQml/qqmlapplicationengine.h>
#include <QtQml/qqmlcontext.h>
#include <QtCore/qtranslator.h>
#include <QtCore/qdatetime.h>
#include <QtCore/qpointer.h>
#include <QtCore/qfile.h>
#include <QtCore/qsettings.h>
#include <QtCore/qdebug.h>
/**
* It allows to grab all debug messages into nootka-log.txt file
*/
void myMessageOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg) {
Q_UNUSED(context)
#else
QFile outFile(QStringLiteral("nootka-log.txt"));
#endif
outFile.open(QIODevice::WriteOnly | QIODevice::Append);
QTextStream ts(&outFile);
}
#if defined (Q_OS_WIN)
// It works under Win
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
#endif
// It mutes QML warnings about connections syntax introduced in Qt 5.15
// TODO when Qt version requirements will rise to 5.15 or above, change syntax and remove that
QLoggingCategory::setFilterRules(QStringLiteral("qt.qml.connections=false"));
qputenv("QT_ANDROID_VOLUME_KEYS", "1"); // Handle volume keys by Qt, lock native Android behavior
logFile = Tandroid::getExternalPath() + QStringLiteral("/nootka-log.txt");
if (QFile::exists(logFile))
QFile::remove(logFile);
qDebug() << "==== NOOTKA LOG =======\n" << QDateTime::currentDateTime().toString();
#else
qputenv("QT_QUICK_CONTROLS_STYLE", ""); // reset style environment var - other styles can cause crashes
SeeLook
committed
QElapsedTimer startElapsed;
startElapsed.start();
QTranslator qtTranslator;
QTranslator nooTranslator;
SeeLook
committed
Tglobals *gl = nullptr;
QPointer<QApplication> a = nullptr;
QQmlApplicationEngine *e = nullptr;
bool firstLoop = true;
SeeLook
committed
#if !defined (Q_OS_ANDROID)
if (a)
delete a;
if (nooObj->resetConfig()) { // delete config file - new Nootka instance will start with first run wizard
SeeLook
committed
QFile f(confFile);
f.remove();
SeeLook
committed
#endif
a = new QApplication(argc, argv);
SeeLook
committed
Tmtr::init(a);
SeeLook
committed
gl = new Tglobals();
gl->path = Tglobals::getInstPath(qApp->applicationDirPath());
confFile = gl->config->fileName();
if (!initCoreLibrary())
return 110;
prepareTranslations(a, qtTranslator, nooTranslator);
if (!loadNootkaFont(a))
return 111;
auto f = a->font();
#if defined (Q_OS_ANDROID)
f.setPixelSize(qRound(static_cast<qreal>(f.pixelSize()) * gl->guiScale()));
auto pal = qApp->palette();
pal.setColor(QPalette::Active, QPalette::Highlight, QColor(0, 160, 160)); // Teal color of highlight for Android
pal.setColor(QPalette::Active, QPalette::Shadow, QColor(144, 144, 144)); // Dark gray for shadow
qApp->setPalette(pal);
#else
f.setPointSizeF(f.pointSizeF() * gl->guiScale());
#endif
a->setFont(f);
a->setWindowIcon(QIcon(Tpath::img("nootka")));
SeeLook
committed
e = new QQmlApplicationEngine;
SeeLook
committed
e->rootContext()->setContextProperty(QStringLiteral("GLOB"), gl);
e->rootContext()->setContextProperty(QStringLiteral("Noo"), nooObj);
e->rootContext()->setContextProperty(QStringLiteral("SOUND"), sound);
SeeLook
committed
bool wasFirstRun = gl->isFirstRun;
TmainHelp* hlp = nullptr; // keep help object live after wizard, Qt deletes it with some delay
SeeLook
committed
if (gl->isFirstRun) {
hlp = new TmainHelp();
e->rootContext()->setContextProperty(QStringLiteral("HELP"), hlp);
e->load(QUrl(QStringLiteral("qrc:/wizard/Wizard.qml")));
if (firstLoop) {
#if defined (Q_OS_ANDROID)
qDebug() << "Nootka wizard launch time" << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]";
#else
QTextStream o(stdout);
o << "\033[01;35m Nootka wizard launch time: " << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]\033[01;00m\n";
exitCode = a->exec();
e->deleteLater(); // Android crashes without a delayed destroy
qApp->quit();
e = new QQmlApplicationEngine;
SeeLook
committed
e->rootContext()->setContextProperty(QStringLiteral("GLOB"), gl);
e->rootContext()->setContextProperty(QStringLiteral("Noo"), nooObj);
e->rootContext()->setContextProperty(QStringLiteral("SOUND"), sound);
SeeLook
committed
gl->isFirstRun = false;
gl->config->setValue(QLatin1String("version"), gl->version);
// TODO: storing current version in settings avoids opening 'about' window next launch
// but if there will be another window - delete line above
qmlRegisterType<TnameItem>("Nootka.Main", 1, 0, "TnameItem");
qmlRegisterType<TmainScoreObject>("Nootka.Main", 1, 0, "TmainScoreObject");
qmlRegisterType<TdialogLoaderObject>("Nootka.Dialogs", 1, 0, "TdialogObject");
#if defined (Q_OS_ANDROID)
qmlRegisterType<TmobileMenu>("Nootka", 1, 0, "TmobileMenu");
#endif
e->load(QUrl(QStringLiteral("qrc:/MainWindow.qml")));
if (firstLoop) {
#if defined (Q_OS_ANDROID)
SeeLook
committed
QString androidArg = Tandroid::getRunArgument();
if (!androidArg.isEmpty())
SeeLook
committed
#else
if (argc > 1)
nooObj->openFile(QString::fromLocal8Bit(argv[argc - 1]));
#endif
}
if (firstLoop && !wasFirstRun) {
SeeLook
committed
#if defined (Q_OS_ANDROID)
qDebug() << "NOOTKA LAUNCH TIME" << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]";
#else
SeeLook
committed
o << "\033[01;35m Nootka launch time: " << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]\033[01;00m\n";
SeeLook
committed
#endif
if (firstLoop && !wasFirstRun) {
// show some dialog when version was changed (news or other info)
if (!TdialogLoaderObject::checkVersion(e->rootObjects().first())) {
// or check updates if no version changed
if (gl->config->value(QLatin1String("Updates/enableUpdates"), true).toBool())
TdialogLoaderObject::updateCheckInBackground();
}
}
firstLoop = false;
SeeLook
committed
delete gl;
if (nooObj->resetConfig()) { // delete config file - new Nootka instance will start with first run wizard
SeeLook
committed
QFile f(confFile);
f.remove();
SeeLook
committed
Tandroid::restartNootka(); // and call Nootka after delay
SeeLook
committed
}
nooObj->setResetConfig(false); // do - while loop doesn't work with Android
qApp->quit(); // HACK: calling QApplication::quick() solves hang on x86 when Qt uses native (usually obsolete) SSL libraries
} while (nooObj->resetConfig());
delete nooObj;
qInstallMessageHandler(0);