main.cpp 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. #include <locale.h>
  2. #include <QGuiApplication>
  3. #include <QApplication>
  4. #include <QFileInfo>
  5. #include <QIcon>
  6. #include <QtQml>
  7. #include <QtWebEngine/qtwebengineglobal.h>
  8. #include <QErrorMessage>
  9. #include "shared/Names.h"
  10. #include "system/SystemComponent.h"
  11. #include "system/UpdateManager.h"
  12. #include "QsLog.h"
  13. #include "Paths.h"
  14. #include "player/CodecsComponent.h"
  15. #include "player/PlayerComponent.h"
  16. #include "player/OpenGLDetect.h"
  17. #include "breakpad/CrashDumps.h"
  18. #include "Version.h"
  19. #include "settings/SettingsComponent.h"
  20. #include "settings/SettingsSection.h"
  21. #include "ui/KonvergoWindow.h"
  22. #include "Globals.h"
  23. #include "ui/ErrorMessage.h"
  24. #include "UniqueApplication.h"
  25. #include "utils/HelperLauncher.h"
  26. #include "utils/Log.h"
  27. #if defined(Q_OS_MAC) || defined(Q_OS_LINUX)
  28. #include "SignalManager.h"
  29. #endif
  30. /////////////////////////////////////////////////////////////////////////////////////////
  31. static void preinitQt()
  32. {
  33. QCoreApplication::setApplicationName(Names::MainName());
  34. QCoreApplication::setApplicationVersion(Version::GetVersionString());
  35. QCoreApplication::setOrganizationDomain("plex.tv");
  36. #ifdef Q_OS_WIN32
  37. QVariant useOpengl = SettingsComponent::readPreinitValue(SETTINGS_SECTION_MAIN, "useOpenGL");
  38. // Warning: this must be the same as the default value as declared in
  39. // the settings_description.json file, or confusion will result.
  40. if (useOpengl.type() != QMetaType::Bool)
  41. useOpengl = false;
  42. if (useOpengl.toBool())
  43. QCoreApplication::setAttribute(Qt::AA_UseDesktopOpenGL);
  44. else
  45. QCoreApplication::setAttribute(Qt::AA_UseOpenGLES);
  46. #endif
  47. }
  48. /////////////////////////////////////////////////////////////////////////////////////////
  49. char** appendCommandLineArguments(int argc, char **argv, const QStringList& args)
  50. {
  51. size_t newSize = (argc + args.length() + 1) * sizeof(char*);
  52. char** newArgv = (char**)calloc(1, newSize);
  53. memcpy(newArgv, argv, (size_t)(argc * sizeof(char*)));
  54. int pos = argc;
  55. for(const QString& str : args)
  56. newArgv[pos++] = qstrdup(str.toUtf8().data());
  57. return newArgv;
  58. }
  59. /////////////////////////////////////////////////////////////////////////////////////////
  60. void ShowLicenseInfo()
  61. {
  62. QFile licenses(":/misc/licenses.txt");
  63. licenses.open(QIODevice::ReadOnly | QIODevice::Text);
  64. QByteArray contents = licenses.readAll();
  65. printf("%.*s\n", contents.size(), contents.data());
  66. }
  67. /////////////////////////////////////////////////////////////////////////////////////////
  68. QStringList g_qtFlags = {"--enable-viewport", "--enable-viewport-meta", "--disable-gpu", "--disable-web-security"};
  69. /////////////////////////////////////////////////////////////////////////////////////////
  70. int main(int argc, char *argv[])
  71. {
  72. try
  73. {
  74. QCommandLineParser parser;
  75. parser.setApplicationDescription("Plex Media Player");
  76. parser.addHelpOption();
  77. parser.addVersionOption();
  78. parser.addOptions({{{"l", "licenses"}, "Show license information"}});
  79. parser.addOptions({{{"a", "from-auto-update"}, "When invoked from auto-update"}});
  80. char **newArgv = appendCommandLineArguments(argc, argv, g_qtFlags);
  81. argc += g_qtFlags.size();
  82. // Suppress SSL related warnings on OSX
  83. // See https://bugreports.qt.io/browse/QTBUG-43173 for more info
  84. //
  85. #ifdef Q_OS_MAC
  86. qputenv("QT_LOGGING_RULES", "qt.network.ssl.warning=false");
  87. #endif
  88. // Qt calls setlocale(LC_ALL, "") in a bunch of places, which breaks
  89. // float/string processing in mpv and ffmpeg.
  90. #ifdef Q_OS_UNIX
  91. qputenv("LC_ALL", "C");
  92. qputenv("LC_NUMERIC", "C");
  93. #endif
  94. detectOpenGLEarly();
  95. preinitQt();
  96. QGuiApplication app(argc, newArgv);
  97. app.setWindowIcon(QIcon(":/images/icon.png"));
  98. // Get the arguments from the app, this is the parsed version of newArgc and newArgv
  99. QStringList args = app.arguments();
  100. // Remove the qt flags above so that our command line parser doesn't get cranky.
  101. for (auto flag : g_qtFlags)
  102. args.removeAll(flag);
  103. // Now parse the command line.
  104. parser.process(args);
  105. if (parser.isSet("licenses"))
  106. {
  107. ShowLicenseInfo();
  108. return EXIT_SUCCESS;
  109. }
  110. // init breakpad.
  111. setupCrashDumper();
  112. UniqueApplication* uniqueApp = new UniqueApplication();
  113. if (!uniqueApp->ensureUnique())
  114. return EXIT_SUCCESS;
  115. #ifdef Q_OS_UNIX
  116. // install signals handlers for proper app closing.
  117. SignalManager signalManager(&app);
  118. Q_UNUSED(signalManager);
  119. #endif
  120. Log::Init();
  121. // Quit app and apply update if we find one.
  122. if (UpdateManager::CheckForUpdates())
  123. {
  124. app.quit();
  125. return 0;
  126. }
  127. detectOpenGLLate();
  128. #ifdef Q_OS_WIN32
  129. initD3DDevice();
  130. #endif
  131. Codecs::preinitCodecs();
  132. // Initialize all the components. This needs to be done
  133. // early since most everything else relies on it
  134. //
  135. ComponentManager::Get().initialize();
  136. // enable remote inspection if we have the correct setting for it.
  137. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "remoteInspector").toBool())
  138. qputenv("QTWEBENGINE_REMOTE_DEBUGGING", "0.0.0.0:9992");
  139. QtWebEngine::initialize();
  140. // start our helper
  141. HelperLauncher::Get().connectToHelper();
  142. // load QtWebChannel so that we can register our components with it.
  143. QQmlApplicationEngine *engine = Globals::Engine();
  144. KonvergoWindow::RegisterClass();
  145. Globals::SetContextProperty("components", &ComponentManager::Get().getQmlPropertyMap());
  146. // the only way to detect if QML parsing fails is to hook to this signal and then see
  147. // if we get a valid object passed to it. Any error messages will be reported on stderr
  148. // but since no normal user should ever see this it should be fine
  149. //
  150. QObject::connect(engine, &QQmlApplicationEngine::objectCreated, [=](QObject* object, const QUrl& url)
  151. {
  152. Q_UNUSED(url);
  153. if (object == nullptr)
  154. throw FatalException(QObject::tr("Failed to parse application engine script."));
  155. KonvergoWindow* window = Globals::MainWindow();
  156. QObject* webChannel = qvariant_cast<QObject*>(window->property("webChannel"));
  157. Q_ASSERT(webChannel);
  158. ComponentManager::Get().setWebChannel(qobject_cast<QWebChannel*>(webChannel));
  159. QObject::connect(uniqueApp, &UniqueApplication::otherApplicationStarted, window, &KonvergoWindow::otherAppFocus);
  160. });
  161. engine->load(QUrl(QStringLiteral("qrc:/ui/webview.qml")));
  162. Log::UpdateLogLevel();
  163. // run our application
  164. int ret = app.exec();
  165. delete uniqueApp;
  166. Globals::EngineDestroy();
  167. Log::Uninit();
  168. return ret;
  169. }
  170. catch (FatalException& e)
  171. {
  172. QLOG_FATAL() << "Unhandled FatalException:" << qPrintable(e.message());
  173. QApplication errApp(argc, argv);
  174. auto msg = new ErrorMessage(e.message(), true);
  175. msg->show();
  176. errApp.exec();
  177. Log::Uninit();
  178. return 1;
  179. }
  180. }