main.cpp 7.2 KB

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