main.cpp 8.2 KB

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