/***************************************************************************
 *   Copyright (C) 2011-2020 by Tomasz Bojczuk                             *
 *   seelook@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 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/>.  *
 ***************************************************************************/


#include <tinitcorelib.h>
#include <tglobals.h>
#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"

#if defined (Q_OS_ANDROID)
  #include <Android/tandroid.h>
  #include "mobile/tmobilemenu.h"
#endif

#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/qelapsedtimer.h>
#include <QtCore/qloggingcategory.h>

#include <QtCore/qdebug.h>


static QString logFile;


/**
 * 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)
  Q_UNUSED(type)

#if defined (Q_OS_ANDROID)
    QFile outFile(logFile);
#else
    QFile outFile(QStringLiteral("nootka-log.txt"));
#endif
    outFile.open(QIODevice::WriteOnly | QIODevice::Append);
    QTextStream ts(&outFile);
    ts << msg << "\n";

}


int main(int argc, char *argv[])
{
#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"));

#if defined (Q_OS_ANDROID)
  qputenv("QT_ANDROID_VOLUME_KEYS", "1"); // Handle volume keys by Qt, lock native Android behavior

  // log to any writable storage
 logFile = Tandroid::getExternalPath() + QStringLiteral("/nootka-log.txt");
  if (QFile::exists(logFile))
    QFile::remove(logFile);
  qInstallMessageHandler(myMessageOutput);
  qDebug() << "==== NOOTKA LOG =======\n" << QDateTime::currentDateTime().toString();
#else
  qputenv("QT_QUICK_CONTROLS_STYLE", ""); // reset style environment var - other styles can cause crashes
#endif

  QElapsedTimer startElapsed;
  startElapsed.start();

  QTranslator qtTranslator;
  QTranslator nooTranslator;
  Tglobals *gl = nullptr;
  QPointer<QApplication> a = nullptr;
  QQmlApplicationEngine *e = nullptr;
  auto nooObj = new TnootkaQML();

  int exitCode;
  bool firstLoop = true;
  QString confFile;
  do {
#if !defined (Q_OS_ANDROID)
    if (a)
      delete a;
    if (nooObj->resetConfig()) { // delete config file - new Nootka instance will start with first run wizard
      QFile f(confFile);
      f.remove();
    }
    nooObj->setResetConfig(false);
#endif

    a = new QApplication(argc, argv);

    Tmtr::init(a);

    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")));

    auto sound = new Tsound();

    e = new QQmlApplicationEngine;
    e->rootContext()->setContextProperty(QStringLiteral("GLOB"), gl);
    e->rootContext()->setContextProperty(QStringLiteral("Noo"), nooObj);
    e->rootContext()->setContextProperty(QStringLiteral("SOUND"), sound);
    bool  wasFirstRun = gl->isFirstRun;
    TmainHelp* hlp = nullptr; // keep help object live after wizard, Qt deletes it with some delay
    if (gl->isFirstRun) {
      hlp = new TmainHelp();
      e->rootContext()->setContextProperty(QStringLiteral("HELP"), hlp);
      nooObj->setQmlEngine(e);
      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";
#endif
      }
      exitCode = a->exec();
      e->deleteLater(); // Android crashes without a delayed destroy
      qApp->quit();
      e = new QQmlApplicationEngine;
      e->rootContext()->setContextProperty(QStringLiteral("GLOB"), gl);
      e->rootContext()->setContextProperty(QStringLiteral("Noo"), nooObj);
      e->rootContext()->setContextProperty(QStringLiteral("SOUND"), sound);
      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
    }
    nooObj->setQmlEngine(e);
    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)
     QString androidArg = Tandroid::getRunArgument();
     if (!androidArg.isEmpty())
       nooObj->openFile(androidArg);
#else
      if (argc > 1)
        nooObj->openFile(QString::fromLocal8Bit(argv[argc - 1]));
#endif
    }
    if (firstLoop && !wasFirstRun) {
#if defined (Q_OS_ANDROID)
      qDebug() << "NOOTKA LAUNCH TIME" << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]";
#else
      QTextStream o(stdout);
      o << "\033[01;35m Nootka launch time: " << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]\033[01;00m\n";
#endif
    }
    sound->init();

    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;
    exitCode = a->exec();

    if (hlp)
      delete hlp;
    delete e;
    delete sound;
    delete gl;
#if defined (Q_OS_ANDROID)
    if (nooObj->resetConfig()) { // delete config file - new Nootka instance will start with first run wizard
        QFile f(confFile);
        f.remove();
        Tandroid::restartNootka(); // and call Nootka after delay
    }
    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
#endif
  } while (nooObj->resetConfig());

  delete nooObj;
  qInstallMessageHandler(0);

  delete a;
  return exitCode;
}