Skip to content
Snippets Groups Projects
main.cpp 16 KiB
Newer Older
SeeLook's avatar
SeeLook committed
/***************************************************************************
 *   Copyright (C) 2011-2025 by Tomasz Bojczuk                             *
 *   seelook@gmail.com                                                     *
SeeLook's avatar
SeeLook committed
 *                                                                         *
 *   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's avatar
SeeLook committed
#include "core/tglobals.h"
#include "core/tinitcorelib.h"
#include "core/tmtr.h"
#include "core/tnootkaqml.h"
#include "core/tpath.h"
#include "dialogs/tdialogloaderobject.h"
#include "help/tmainhelp.h"
#include "main/tgotit.h"
#include "main/tmainscoreobject.h"
#include "main/tnameitem.h"
SeeLook's avatar
SeeLook committed
#include "sound/tsound.h"
#if defined(Q_OS_ANDROID)
#include "mobile/tmobilemenu.h"
#include <Android/tandroid.h>
#include "nootini/taudioanalyzeitem.h"
#include <QtCore/qcommandlineparser.h>
#include <QtCore/qdatetime.h>
#include <QtCore/qdir.h>
SeeLook's avatar
SeeLook committed
#include <QtCore/qelapsedtimer.h>
#include <QtCore/qfile.h>
#include <QtCore/qloggingcategory.h>
#include <QtCore/qpointer.h>
#include <QtCore/qsettings.h>
#include <QtCore/qtranslator.h>
#include <QtGui/qicon.h>
#include <QtGui/qpalette.h>
#include <QtQml/qqmlapplicationengine.h>
#include <QtQml/qqmlcontext.h>
#include <QtWidgets/qapplication.h>
#include <QtCore/qdebug.h>

static QString logFile;
SeeLook's avatar
SeeLook committed

/**
 * 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";
SeeLook's avatar
SeeLook committed
int main(int argc, char *argv[])
#if defined(Q_OS_WIN)
    // It works under Win
    QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
#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
    if (Tandroid::hasWriteAccess()) {
        logFile = Tandroid::getExternalPath() + QStringLiteral("/nootka-log.txt");
        Tandroid::askForWriteAcces();
        if (QFile::exists(logFile))
            QFile::remove(logFile);
        qInstallMessageHandler(myMessageOutput);
        qDebug() << "==== NOOTKA LOG =======\n" << QDateTime::currentDateTime().toString();
    }
SeeLook's avatar
SeeLook committed
    qputenv("QT_QUICK_CONTROLS_STYLE", "Basic"); // reset style environment var - other styles can cause crashes
    QElapsedTimer startElapsed;
    startElapsed.start();

    QTranslator qtTranslator;
    QTranslator nooTranslator;
    Tglobals *glob = nullptr;
    QPointer<QApplication> a = nullptr;
    QQmlApplicationEngine *engine = nullptr;
    TnootkaQML *nooQML = nullptr;
    bool resetConfig = false;

    int exitCode;
    bool firstLoop = true;
    QString confFile;
    do {
        resetConfig = false;
        nooQML = new TnootkaQML();
#if !defined(Q_OS_ANDROID)
        if (a)
            delete a;
        a = new QApplication(argc, argv);

        Tmtr::init(a);

        glob = new Tglobals();
        glob->path = Tglobals::getInstPath(qApp->applicationDirPath());
        confFile = glob->config->fileName();
        if (!initCoreLibrary())
            return 110;
        prepareTranslations(a, qtTranslator, nooTranslator);
        if (!loadNootkaFont(a))
            return 111;

        auto f = a->font();
#if defined(Q_OS_ANDROID)
SeeLook's avatar
SeeLook committed
        f.setPixelSize(nooQML->factor());
        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(120, 120, 120)); // Dark gray for shadow
        pal.setColor(QPalette::Active, QPalette::Window, QColor(250, 250, 250)); // Almost white for windows
        pal.setColor(QPalette::Active, QPalette::Base, QColor(255, 255, 255)); // White base
        pal.setColor(QPalette::Active, QPalette::Button, QColor(240, 240, 240)); // Very light gray for button
        pal.setColor(QPalette::Active, QPalette::AlternateBase, QColor(245, 255, 255)); // Very light teal for alternate base
        qApp->setPalette(pal);
#elif defined(Q_OS_WIN)
        QSettings accent(QStringLiteral("HKEY_USERS\\.DEFAULT\\Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Accent"), QSettings::NativeFormat);
        if (accent.contains(QLatin1String("StartColorMenu"))) {
            int color = accent.value(QLatin1String("StartColorMenu")).toInt();
            int r = color & 0xff;
            int g = (color >> 8) & 0xff;
            int b = (color >> 16) & 0xff;
            auto pal = qApp->palette();
            QColor c(r, g, b);
            pal.setColor(QPalette::Active, QPalette::Highlight, c.lighter(150));
            qApp->setPalette(pal);
        }
SeeLook's avatar
SeeLook committed
        f.setPointSizeF(f.pointSizeF() * glob->guiScale());
        f.setPointSizeF(f.pointSizeF() * glob->guiScale());
        a->setFont(f);

        a->setWindowIcon(QIcon(Tpath::img("nootka")));

        auto sound = new Tsound();

        engine = new QQmlApplicationEngine;

        qmlRegisterSingletonType<Tglobals>("Nootka", 1, 0, "GLOB",
                                          [&](QQmlEngine*, QJSEngine*)->QObject* {
                                          });

        qmlRegisterSingletonType<TnootkaQML>("Nootka", 1, 0, "NOO",
                                           [&](QQmlEngine*, QJSEngine*)->QObject* {
                                           });

        qmlRegisterSingletonType<Tsound>("Nootka", 1, 0, "SOUND",
                                             [&](QQmlEngine*, QJSEngine*)->QObject* {
                                                 return sound;
                                             });

        bool wasFirstRun = glob->isFirstRun;
        TmainHelp *hlp = nullptr; // keep help object live after wizard, Qt deletes it with some delay
        if (glob->isFirstRun) {
            hlp = new TmainHelp();
            engine->rootContext()->setContextProperty(QStringLiteral("HELP"), hlp);
            nooQML->setQmlEngine(engine);
            engine->load(QUrl(QStringLiteral("qrc:/wizard/Wizard.qml")));
            if (firstLoop) {
#if defined(Q_OS_ANDROID)
                qDebug() << "Nootka wizard launch time" << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]";
                QTextStream o(stdout);
                o << "\033[01;35m Nootka wizard launch time: " << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]\033[01;00m\n";
            }
            exitCode = a->exec();
            glob->isFirstRun = false;
            glob->config->setValue(QLatin1String("version"), glob->version);
            // TODO: storing current version in settings avoids opening 'about' window next launch
            //       but if there will be another window - delete line above
        }
        nooQML->setQmlEngine(engine);
        qmlRegisterType<TgotIt>("Nootka.Main", 1, 0, "TgotIt");
        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");
        engine->load(QUrl(QStringLiteral("qrc:/MainWindow.qml")));
SeeLook's avatar
SeeLook committed

#if !defined(Q_OS_ANDROID)
        if (engine->rootObjects().isEmpty()) {
            QTextStream o(stdout);
            o << "\033[0;31m Something went wrong and Nootka was not able to launch.\033[01;00m\n";
            return 121;
        }
        if (firstLoop && !wasFirstRun) {
#if defined(Q_OS_ANDROID)
            qDebug() << "NOOTKA LAUNCH TIME" << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]";
            QTextStream o(stdout);
            o << "\033[01;35m Nootka launch time: " << startElapsed.nsecsElapsed() / 1000000.0 << " [ms]\033[01;00m\n";
        if (firstLoop) {
#if defined(Q_OS_ANDROID)
            QString androidArg = Tandroid::getRunArgument();
            if (!androidArg.isEmpty())
SeeLook's avatar
SeeLook committed
                nooQML->openFile(androidArg);
            if (argc > 1) {
                QCommandLineParser cmd;
                auto helpOpt = cmd.addHelpOption();
                QCommandLineOption dumpOpt(QStringList() << QStringLiteral("dump-dir") << QStringLiteral("d"),
                                           QStringLiteral("Full path to directory where audio data used to pitch detection will be dumped.\n"
                                                          "Dumped files can be further analyzed with 'nootini' option.\n"),
                                           QStringLiteral("existing dir"));
                cmd.addOption(dumpOpt);
                QCommandLineOption nootiniOpt(QStringList() << QStringLiteral("nootini") << QStringLiteral("n"),
                                              QStringLiteral("Launch Nootka in audio analyze mode. Nootini: (Nootka + Tartini)\n"));
                cmd.addOption(nootiniOpt);
                QCommandLineOption audioFileOpt(QStringList() << QStringLiteral("audio-file") << QStringLiteral("a"),
                                                QStringLiteral("Audio file to analyze. Only raw files dumped by Nootka are supported.\n"),
                                                QStringLiteral("wav or raw audio"));
                cmd.addOption(audioFileOpt);
                QCommandLineOption tempoOpt(QStringList() << QStringLiteral("tempo") << QStringLiteral("t"),
                                            QStringLiteral("Tempo of given audio file.\n"),
                                            QStringLiteral("bpm"));
                cmd.addOption(tempoOpt);
                QCommandLineOption instrOpt(QStringList() << QStringLiteral("instrument") << QStringLiteral("i"),
                                            QStringLiteral("Instrument to use. See Tinstrument class (tinstrument.h)\n"),
                                            QStringLiteral("number"));
                cmd.addOption(instrOpt);
                QCommandLineOption quantOpt(QStringList() << QStringLiteral("quantization") << QStringLiteral("q"),
                                            QStringLiteral("Quantization (round to): sixteenths (6) or eights (12)\n"),
                                            QStringLiteral("6 or 12"));
                cmd.addOption(quantOpt);
                QCommandLineOption minVolOpt(QStringList() << QStringLiteral("min-volume") << QStringLiteral("m"),
                                             QStringLiteral("Minimal note volume to be pitch detected in percents.\n"),
                                             QStringLiteral("20-80"));
                cmd.addOption(minVolOpt);
                QCommandLineOption clefOpt(QStringList() << QStringLiteral("clef") << QStringLiteral("c"),
                                           QStringLiteral("Clef for score. See Tclef class (tclef.h)\n"),
                                           QStringLiteral("number (power of 2)"));
                cmd.addOption(clefOpt);

                /** Option below is handled internally by @p TnootkaQML. */
                cmd.addOptions({{QStringLiteral("no-version"), QStringLiteral("Do not display app version.\n")}});

                cmd.parse(a->arguments());
                if (cmd.isSet(helpOpt))
                    cmd.showHelp();

                if (cmd.isSet(dumpOpt))
                    SOUND->changeDumpPath(cmd.value(dumpOpt));
                else if (cmd.isSet(nootiniOpt)) {
                    qmlRegisterType<TaudioAnalyzeItem>("Nootka", 1, 0, "TaudioAnalyzeItem");
                    QMetaObject::invokeMethod(engine->rootObjects().first(), "audioAnalyze");
                    if (cmd.isSet(instrOpt)) {
                        int instr = cmd.value(instrOpt).toInt();
                        GLOB->setTransposition(Tinstrument(static_cast<Tinstrument::Etype>(instr)).transposition());
                        GLOB->setInstrument(instr);
                    }
                    if (cmd.isSet(quantOpt))
                        SOUND->setQuantization(cmd.value(quantOpt).toInt());
                    if (cmd.isSet(tempoOpt))
                        SOUND->setTempo(cmd.value(tempoOpt).toInt());
                    if (cmd.isSet(minVolOpt))
                        GLOB->setMinVolume(static_cast<qreal>(qBound(20, cmd.value(minVolOpt).toInt(), 80)) / 100.0);
                    if (cmd.isSet(audioFileOpt))
                        TaudioAnalyzeItem::processAudioFile(cmd.value(audioFileOpt));
                    if (cmd.isSet(clefOpt))
                        GLOB->setClefType(cmd.value(clefOpt).toInt());
                } else
                    nooQML->openFile(QString::fromLocal8Bit(argv[argc - 1]));
        if (firstLoop && !wasFirstRun && !engine->rootObjects().isEmpty()) {
            // show dialog with changes when version was changed
            if (!TdialogLoaderObject::checkVersion(engine->rootObjects().first())) {
                // or check to dispay support dialog
                if (!TdialogLoaderObject::checkForSupport(engine->rootObjects().first())) {
                    // or check for updates if any of above
                    if (glob->config->value(QLatin1String("Updates/enableUpdates"), true).toBool())
                        TdialogLoaderObject::updateCheckInBackground();
                }
            }
        firstLoop = false;
        exitCode = a->exec();

        resetConfig = nooQML->resetConfig();

        delete engine;
        /** by deleting @p QQmlEngine engine all singletons: @p sound, @p glob, @p nooQML are also deleted */

#if defined(Q_OS_ANDROID)
        if (resetConfig) { // delete config file - new Nootka instance will start with first run wizard
            // QFile f(confFile); // TODO: check this and remove
            // f.remove();
            Tandroid::restartNootka(); // and call Nootka after delay
        }
        qApp->quit(); // HACK: calling QApplication::quick() solves hang on x86 when Qt uses native (usually obsolete) SSL libraries
#else
        if (resetConfig) { // delete config file - new Nootka instance will start with first run wizard
            qDebug() << "Reset configuration" << confFile;
            QFile f(confFile);
            f.remove();
            auto exerciseFile = QDir::toNativeSeparators(QFileInfo(confFile).absolutePath() + QLatin1String("/exercise2.noo"));
            if (QFileInfo(exerciseFile).exists())
                QFile::remove(exerciseFile);
            // also clean exercise from Nootka versions up to 1.7.X
            exerciseFile = QDir::toNativeSeparators(QFileInfo(confFile).absolutePath() + QLatin1String("/exercise.noo"));
            if (QFileInfo(exerciseFile).exists())
                QFile::remove(exerciseFile);
        }
    } while (resetConfig);
    qInstallMessageHandler(nullptr);
    delete a;
    return exitCode;
SeeLook's avatar
SeeLook committed
}