#include "amnezia_application.h" #include #include #include #include #include #include #include #include #include #include #include #include #include "logger.h" #include "ui/controllers/pageController.h" #include "ui/models/installedAppsModel.h" #include "version.h" #include "platforms/ios/QRCodeReaderBase.h" #include "protocols/qml_register_protocols.h" AmneziaApplication::AmneziaApplication(int &argc, char *argv[]) : AMNEZIA_BASE_CLASS(argc, argv) { setQuitOnLastWindowClosed(false); // Fix config file permissions #if defined(Q_OS_LINUX) && !defined(Q_OS_ANDROID) { QSettings s(ORGANIZATION_NAME, APPLICATION_NAME); s.setValue("permFixed", true); } QString configLoc1 = QStandardPaths::standardLocations(QStandardPaths::ConfigLocation).first() + "/" + ORGANIZATION_NAME + "/" + APPLICATION_NAME + ".conf"; QFile::setPermissions(configLoc1, QFileDevice::ReadOwner | QFileDevice::WriteOwner); QString configLoc2 = QStandardPaths::standardLocations(QStandardPaths::ConfigLocation).first() + "/" + ORGANIZATION_NAME + "/" + APPLICATION_NAME + "/" + APPLICATION_NAME + ".conf"; QFile::setPermissions(configLoc2, QFileDevice::ReadOwner | QFileDevice::WriteOwner); #endif m_settings = std::shared_ptr(new Settings); m_nam = new QNetworkAccessManager(this); } AmneziaApplication::~AmneziaApplication() { m_vpnConnectionThread.quit(); m_vpnConnectionThread.wait(3000); if (m_engine) { QObject::disconnect(m_engine, 0, 0, 0); delete m_engine; } } void AmneziaApplication::init() { m_engine = new QQmlApplicationEngine; const QUrl url(QStringLiteral("qrc:/ui/qml/main2.qml")); QObject::connect( m_engine, &QQmlApplicationEngine::objectCreated, this, [url](QObject *obj, const QUrl &objUrl) { if (!obj && url == objUrl) QCoreApplication::exit(-1); }, Qt::QueuedConnection); m_engine->rootContext()->setContextProperty("Debug", &Logger::Instance()); m_vpnConnection.reset(new VpnConnection(m_settings)); m_vpnConnection->moveToThread(&m_vpnConnectionThread); m_vpnConnectionThread.start(); m_coreController.reset(new CoreController(m_vpnConnection, m_settings, m_engine)); m_engine->addImportPath("qrc:/ui/qml/Modules/"); m_engine->load(url); m_coreController->setQmlRoot(); bool enabled = m_settings->isSaveLogs(); #ifndef Q_OS_ANDROID if (enabled) { if (!Logger::init(false)) { qWarning() << "Initialization of debug subsystem failed"; } } #endif Logger::setServiceLogsEnabled(enabled); #ifdef Q_OS_WIN //TODO if (m_parser.isSet("a")) m_coreController->pageController()->showOnStartup(); else emit m_coreController->pageController()->raiseMainWindow(); #else m_coreController->pageController()->showOnStartup(); #endif // Android TextArea clipboard workaround // Text from TextArea always has "text/html" mime-type: // /qt/6.6.1/Src/qtdeclarative/src/quick/items/qquicktextcontrol.cpp:1865 // Next, html is created for this mime-type: // /qt/6.6.1/Src/qtdeclarative/src/quick/items/qquicktextcontrol.cpp:1885 // And this html goes to the Androids clipboard, i.e. text from TextArea is always copied as richText: // /qt/6.6.1/Src/qtbase/src/plugins/platforms/android/androidjniclipboard.cpp:46 // So we catch all the copies to the clipboard and clear them from "text/html" #ifdef Q_OS_ANDROID connect(QGuiApplication::clipboard(), &QClipboard::dataChanged, []() { auto clipboard = QGuiApplication::clipboard(); if (clipboard->mimeData()->hasHtml()) { clipboard->setText(clipboard->text()); } }); #endif } void AmneziaApplication::registerTypes() { qRegisterMetaType("ServerCredentials"); qRegisterMetaType("DockerContainer"); qRegisterMetaType("TransportProto"); qRegisterMetaType("Proto"); qRegisterMetaType("ServiceType"); declareQmlProtocolEnum(); declareQmlContainerEnum(); qmlRegisterType("QRCodeReader", 1, 0, "QRCodeReader"); m_containerProps.reset(new ContainerProps()); qmlRegisterSingletonInstance("ContainerProps", 1, 0, "ContainerProps", m_containerProps.get()); m_protocolProps.reset(new ProtocolProps()); qmlRegisterSingletonInstance("ProtocolProps", 1, 0, "ProtocolProps", m_protocolProps.get()); qmlRegisterSingletonType(QUrl("qrc:/ui/qml/Filters/ContainersModelFilters.qml"), "ContainersModelFilters", 1, 0, "ContainersModelFilters"); qmlRegisterType("InstalledAppsModel", 1, 0, "InstalledAppsModel"); Vpn::declareQmlVpnConnectionStateEnum(); PageLoader::declareQmlPageEnum(); } void AmneziaApplication::loadFonts() { QQuickStyle::setStyle("Basic"); QFontDatabase::addApplicationFont(":/fonts/pt-root-ui_vf.ttf"); } bool AmneziaApplication::parseCommands() { m_parser.setApplicationDescription(APPLICATION_NAME); m_parser.addHelpOption(); m_parser.addVersionOption(); QCommandLineOption c_autostart { { "a", "autostart" }, "System autostart" }; m_parser.addOption(c_autostart); QCommandLineOption c_cleanup { { "c", "cleanup" }, "Cleanup logs" }; m_parser.addOption(c_cleanup); m_parser.process(*this); if (m_parser.isSet(c_cleanup)) { Logger::cleanUp(); QTimer::singleShot(100, this, [this] { quit(); }); exec(); return false; } return true; } #if !defined(Q_OS_ANDROID) && !defined(Q_OS_IOS) void AmneziaApplication::startLocalServer() { const QString serverName("AmneziaVPNInstance"); QLocalServer::removeServer(serverName); QLocalServer *server = new QLocalServer(this); server->listen(serverName); QObject::connect(server, &QLocalServer::newConnection, this, [server, this]() { if (server) { QLocalSocket *clientConnection = server->nextPendingConnection(); clientConnection->deleteLater(); } emit m_coreController->pageController()->raiseMainWindow(); //TODO }); } #endif QQmlApplicationEngine *AmneziaApplication::qmlEngine() const { return m_engine; } QNetworkAccessManager *AmneziaApplication::networkManager() { return m_nam; } QClipboard *AmneziaApplication::getClipboard() { return this->clipboard(); }