PlayerComponent.cpp 50 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518
  1. #include "PlayerComponent.h"
  2. #include <QString>
  3. #include <Qt>
  4. #include <QDir>
  5. #include <QCoreApplication>
  6. #include <QGuiApplication>
  7. #include "display/DisplayComponent.h"
  8. #include "settings/SettingsComponent.h"
  9. #include "system/SystemComponent.h"
  10. #include "utils/Utils.h"
  11. #include "utils/Log.h"
  12. #include "ComponentManager.h"
  13. #include "settings/SettingsSection.h"
  14. #include "PlayerQuickItem.h"
  15. #include "input/InputComponent.h"
  16. #include "QsLog.h"
  17. #include <math.h>
  18. #include <string.h>
  19. #include <shared/Paths.h>
  20. #if !defined(Q_OS_WIN)
  21. #include <unistd.h>
  22. #endif
  23. #if !defined(Q_OS_WIN)
  24. #include <unistd.h>
  25. #endif
  26. #ifdef TARGET_RPI
  27. #include <bcm_host.h>
  28. #include <interface/vmcs_host/vcgencmd.h>
  29. #endif
  30. ///////////////////////////////////////////////////////////////////////////////////////////////////
  31. static void wakeup_cb(void *context)
  32. {
  33. PlayerComponent *player = (PlayerComponent *)context;
  34. emit player->onMpvEvents();
  35. }
  36. ///////////////////////////////////////////////////////////////////////////////////////////////////
  37. PlayerComponent::PlayerComponent(QObject* parent)
  38. : ComponentBase(parent), m_state(State::finished), m_paused(false), m_playbackActive(false),
  39. m_windowVisible(false), m_videoPlaybackActive(false), m_inPlayback(false), m_playbackCanceled(false),
  40. m_bufferingPercentage(100), m_lastBufferingPercentage(-1),
  41. m_lastPositionUpdate(0.0), m_playbackAudioDelay(0),
  42. m_window(nullptr), m_mediaFrameRate(0),
  43. m_restoreDisplayTimer(this), m_reloadAudioTimer(this),
  44. m_streamSwitchImminent(false), m_doAc3Transcoding(false),
  45. m_videoRectangle(-1, -1, -1, -1)
  46. {
  47. qmlRegisterType<PlayerQuickItem>("Konvergo", 1, 0, "MpvVideo"); // deprecated name
  48. qmlRegisterType<PlayerQuickItem>("Konvergo", 1, 0, "KonvergoVideo");
  49. m_restoreDisplayTimer.setSingleShot(true);
  50. connect(&m_restoreDisplayTimer, &QTimer::timeout, this, &PlayerComponent::onRestoreDisplay);
  51. connect(&DisplayComponent::Get(), &DisplayComponent::refreshRateChanged, this, &PlayerComponent::onRefreshRateChange);
  52. m_reloadAudioTimer.setSingleShot(true);
  53. connect(&m_reloadAudioTimer, &QTimer::timeout, this, &PlayerComponent::updateAudioDevice);
  54. }
  55. /////////////////////////////////////////////////////////////////////////////////////////
  56. void PlayerComponent::componentPostInitialize()
  57. {
  58. InputComponent::Get().registerHostCommand("player", this, "userCommand");
  59. }
  60. ///////////////////////////////////////////////////////////////////////////////////////////////////
  61. PlayerComponent::~PlayerComponent()
  62. {
  63. if (m_mpv)
  64. mpv_set_wakeup_callback(m_mpv, nullptr, nullptr);
  65. }
  66. ///////////////////////////////////////////////////////////////////////////////////////////////////
  67. bool PlayerComponent::componentInitialize()
  68. {
  69. m_mpv = mpv::qt::Handle::FromRawHandle(mpv_create());
  70. if (!m_mpv)
  71. throw FatalException(tr("Failed to load mpv."));
  72. mpv_request_log_messages(m_mpv, "terminal-default");
  73. mpv::qt::set_property(m_mpv, "msg-level", "all=v");
  74. mpv::qt::set_property(m_mpv, "config", "yes");
  75. mpv::qt::set_property(m_mpv, "config-dir", Paths::dataDir());
  76. mpv_set_wakeup_callback(m_mpv, wakeup_cb, this);
  77. // Disable native OSD if mpv_command_string() is used.
  78. mpv::qt::set_property(m_mpv, "osd-level", "0");
  79. // This forces the player not to rebase playback time to 0 with mkv. We
  80. // require this, because mkv transcoding lets files start at times other
  81. // than 0, and web-client expects that we return these times unchanged.
  82. mpv::qt::set_property(m_mpv, "demuxer-mkv-probe-start-time", false);
  83. // Upstream mpv sets this to "auto", which disables probing for HLS (at least),
  84. // in order to speed up playback start. The situation is more complex in PMP
  85. // due to us wanting to use system codecs, so always enable this.
  86. mpv::qt::set_property(m_mpv, "demuxer-lavf-probe-info", true);
  87. // Just discard audio output if no audio device could be opened. This gives
  88. // us better flexibility how to react to such errors (instead of just
  89. // aborting playback immediately).
  90. mpv::qt::set_property(m_mpv, "audio-fallback-to-null", "yes");
  91. // Do not let the decoder downmix (better customization for us).
  92. mpv::qt::set_property(m_mpv, "ad-lavc-downmix", false);
  93. // Make it load the hwdec interop, so hwdec can be enabled at runtime.
  94. mpv::qt::set_property(m_mpv, "hwdec-preload", "auto");
  95. // User-visible application name used by some audio APIs (at least PulseAudio).
  96. mpv::qt::set_property(m_mpv, "audio-client-name", QCoreApplication::applicationName());
  97. // User-visible stream title used by some audio APIs (at least PulseAudio and wasapi).
  98. mpv::qt::set_property(m_mpv, "title", QCoreApplication::applicationName());
  99. mpv::qt::set_property(m_mpv, "tls-verify", "yes");
  100. #if !defined(Q_OS_WIN) && !defined(Q_OS_MAC)
  101. QList<QByteArray> list;
  102. list << "/etc/ssl/certs/ca-certificates.crt"
  103. << "/etc/pki/tls/certs/ca-bundle.crt"
  104. << "/usr/share/ssl/certs/ca-bundle.crt"
  105. << "/usr/local/share/certs/ca-root-nss.crt"
  106. << "/etc/ssl/cert.pem"
  107. << "/usr/share/curl/curl-ca-bundle.crt"
  108. << "/usr/local/share/curl/curl-ca-bundle.crt";
  109. bool success = false;
  110. for (auto path : list)
  111. {
  112. if (access(path.data(), R_OK) == 0) {
  113. mpv::qt::set_property(m_mpv, "tls-ca-file", path.data());
  114. success = true;
  115. break;
  116. }
  117. }
  118. if (!success)
  119. throw FatalException(tr("Failed to locate CA bundle."));
  120. #endif
  121. // Apply some low-memory settings on RPI, which is relatively memory-constrained.
  122. #ifdef TARGET_RPI
  123. // The backbuffer makes seeking back faster (without having to do a HTTP-level seek)
  124. mpv::qt::set_property(m_mpv, "cache-backbuffer", 10 * 1024); // KB
  125. // The demuxer queue is used for the readahead, and also for dealing with badly
  126. // interlaved audio/video. Setting it too low increases sensitivity to network
  127. // issues, and could cause playback failure with "bad" files.
  128. mpv::qt::set_property(m_mpv, "demuxer-max-bytes", 50 * 1024 * 1024); // bytes
  129. // Specifically for enabling mpeg4.
  130. mpv::qt::set_property(m_mpv, "hwdec-codecs", "all");
  131. // Do not use exact seeks by default. (This affects the start position in the "loadfile"
  132. // command in particular. We override the seek mode for normal "seek" commands.)
  133. mpv::qt::set_property(m_mpv, "hr-seek", "no");
  134. // Force vo_rpi to fullscreen.
  135. mpv::qt::set_property(m_mpv, "fullscreen", true);
  136. #endif
  137. if (mpv_initialize(m_mpv) < 0)
  138. throw FatalException(tr("Failed to initialize mpv."));
  139. mpv_observe_property(m_mpv, 0, "pause", MPV_FORMAT_FLAG);
  140. mpv_observe_property(m_mpv, 0, "core-idle", MPV_FORMAT_FLAG);
  141. mpv_observe_property(m_mpv, 0, "cache-buffering-state", MPV_FORMAT_INT64);
  142. mpv_observe_property(m_mpv, 0, "playback-time", MPV_FORMAT_DOUBLE);
  143. mpv_observe_property(m_mpv, 0, "vo-configured", MPV_FORMAT_FLAG);
  144. mpv_observe_property(m_mpv, 0, "duration", MPV_FORMAT_DOUBLE);
  145. mpv_observe_property(m_mpv, 0, "audio-device-list", MPV_FORMAT_NODE);
  146. mpv_observe_property(m_mpv, 0, "video-dec-params", MPV_FORMAT_NODE);
  147. // Setup a hook with the ID 1, which is run during the file is loaded.
  148. // Used to delay playback start for display framerate switching.
  149. // (See handler in handleMpvEvent() for details.)
  150. mpv::qt::command(m_mpv, QStringList() << "hook-add" << "on_load" << "1" << "0");
  151. // Setup a hook with the ID 2, which is run at a certain stage during loading.
  152. // We use it to initialize stream selections and to probe the codecs.
  153. mpv::qt::command(m_mpv, QStringList() << "hook-add" << "on_preloaded" << "2" << "0");
  154. updateAudioDeviceList();
  155. setAudioConfiguration();
  156. updateSubtitleSettings();
  157. updateVideoSettings();
  158. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_VIDEO), &SettingsSection::valuesUpdated,
  159. this, &PlayerComponent::updateVideoSettings);
  160. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_SUBTITLES), &SettingsSection::valuesUpdated,
  161. this, &PlayerComponent::updateSubtitleSettings);
  162. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_AUDIO), &SettingsSection::valuesUpdated,
  163. this, &PlayerComponent::setAudioConfiguration);
  164. initializeCodecSupport();
  165. Codecs::initCodecs();
  166. QString codecInfo;
  167. for (auto codec : Codecs::getCachedCodecList())
  168. {
  169. if (codec.present)
  170. {
  171. if (codecInfo.size())
  172. codecInfo += " ";
  173. codecInfo += codec.driver;
  174. if (codec.type == CodecType::Encoder)
  175. codecInfo += "(enc)";
  176. }
  177. }
  178. QLOG_INFO() << "Present codecs:" << qPrintable(codecInfo);
  179. connect(this, &PlayerComponent::onMpvEvents, this, &PlayerComponent::handleMpvEvents, Qt::QueuedConnection);
  180. emit onMpvEvents();
  181. return true;
  182. }
  183. ///////////////////////////////////////////////////////////////////////////////////////////////////
  184. void PlayerComponent::setVideoRectangle(int x, int y, int w, int h)
  185. {
  186. QRect rc(x, y, w, h);
  187. if (rc != m_videoRectangle)
  188. {
  189. m_videoRectangle = rc;
  190. emit onVideoRecangleChanged();
  191. }
  192. }
  193. ///////////////////////////////////////////////////////////////////////////////////////////////////
  194. void PlayerComponent::setQtQuickWindow(QQuickWindow* window)
  195. {
  196. PlayerQuickItem* video = window->findChild<PlayerQuickItem*>("video");
  197. if (!video)
  198. throw FatalException(tr("Failed to load video element."));
  199. video->initMpv(this);
  200. }
  201. ///////////////////////////////////////////////////////////////////////////////////////////////////
  202. void PlayerComponent::setWindow(QQuickWindow* window)
  203. {
  204. QString vo = "opengl-cb";
  205. #ifdef TARGET_RPI
  206. window->setFlags(Qt::FramelessWindowHint);
  207. vo = "rpi";
  208. #endif
  209. m_window = window;
  210. if (!window)
  211. return;
  212. QString forceVo = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "debug.force_vo").toString();
  213. if (forceVo.size())
  214. vo = forceVo;
  215. mpv::qt::set_property(m_mpv, "vo", vo);
  216. if (vo == "opengl-cb")
  217. setQtQuickWindow(window);
  218. }
  219. ///////////////////////////////////////////////////////////////////////////////////////////////////
  220. bool PlayerComponent::load(const QString& url, const QVariantMap& options, const QVariantMap &metadata, const QString& audioStream , const QString& subtitleStream)
  221. {
  222. stop();
  223. queueMedia(url, options, metadata, audioStream, subtitleStream);
  224. return true;
  225. }
  226. ///////////////////////////////////////////////////////////////////////////////////////////////////
  227. static QString ConvertPlexDirectURL(const QString& host)
  228. {
  229. if (!host.endsWith(".plex.direct"))
  230. return host;
  231. QString substr = host.left(host.indexOf('.'));
  232. substr.replace('-', '.');
  233. return substr;
  234. }
  235. ///////////////////////////////////////////////////////////////////////////////////////////////////
  236. void PlayerComponent::queueMedia(const QString& url, const QVariantMap& options, const QVariantMap &metadata, const QString& audioStream, const QString& subtitleStream)
  237. {
  238. InputComponent::Get().cancelAutoRepeat();
  239. m_mediaFrameRate = metadata["frameRate"].toFloat(); // returns 0 on failure
  240. m_serverMediaInfo = metadata["media"].toMap();
  241. updateVideoSettings();
  242. QUrl qurl = url;
  243. QString host = qurl.host();
  244. qurl.setHost(ConvertPlexDirectURL(host));
  245. QVariantList command;
  246. command << "loadfile" << qurl.toString();
  247. command << "append-play"; // if nothing is playing, play it now, otherwise just enqueue it
  248. QVariantMap extraArgs;
  249. quint64 startMilliseconds = options["startMilliseconds"].toLongLong();
  250. if (startMilliseconds != 0)
  251. extraArgs.insert("start", "+" + QString::number(startMilliseconds / 1000.0));
  252. // we're going to select these streams later, in the preloaded hook
  253. extraArgs.insert("aid", "no");
  254. extraArgs.insert("sid", "no");
  255. m_currentSubtitleStream = subtitleStream;
  256. m_currentAudioStream = audioStream;
  257. if (metadata["type"] == "music")
  258. extraArgs.insert("vid", "no");
  259. extraArgs.insert("pause", options["autoplay"].toBool() ? "no" : "yes");
  260. QString userAgent = metadata["headers"].toMap()["User-Agent"].toString();
  261. if (userAgent.size())
  262. extraArgs.insert("user-agent", userAgent);
  263. // Make sure the list of requested codecs is reset.
  264. extraArgs.insert("ad", "");
  265. extraArgs.insert("vd", "");
  266. extraArgs.insert("stream-lavf-o", "verifyhost=" + host);
  267. command << extraArgs;
  268. mpv::qt::command(m_mpv, command);
  269. }
  270. /////////////////////////////////////////////////////////////////////////////////////////
  271. void PlayerComponent::streamSwitch()
  272. {
  273. m_streamSwitchImminent = true;
  274. }
  275. ///////////////////////////////////////////////////////////////////////////////////////////////////
  276. bool PlayerComponent::switchDisplayFrameRate()
  277. {
  278. QLOG_DEBUG() << "Video framerate:" << m_mediaFrameRate << "fps";
  279. if (!SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "refreshrate.auto_switch").toBool())
  280. {
  281. QLOG_DEBUG() << "Not switching refresh-rate (disabled by settings).";
  282. return false;
  283. }
  284. bool fs = SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool();
  285. #if KONVERGO_OPENELEC
  286. fs = true;
  287. #endif
  288. if (!fs)
  289. {
  290. QLOG_DEBUG() << "Not switching refresh-rate (not in fullscreen mode).";
  291. return false;
  292. }
  293. if (m_mediaFrameRate < 1)
  294. {
  295. QLOG_DEBUG() << "Not switching refresh-rate (no known video framerate).";
  296. return false;
  297. }
  298. // Make sure a timer started by the previous file ending isn't accidentally
  299. // still in-flight. It could switch the display back after we've switched.
  300. m_restoreDisplayTimer.stop();
  301. DisplayComponent* display = &DisplayComponent::Get();
  302. if (!display->switchToBestVideoMode(m_mediaFrameRate))
  303. {
  304. QLOG_DEBUG() << "Switching refresh-rate failed or unnecessary.";
  305. return false;
  306. }
  307. // Make sure settings dependent on the display refresh rate are updated properly.
  308. updateVideoSettings();
  309. return true;
  310. }
  311. ///////////////////////////////////////////////////////////////////////////////////////////////////
  312. void PlayerComponent::onRestoreDisplay()
  313. {
  314. // If the player will in fact start another file (or is playing one), don't restore.
  315. if (mpv::qt::get_property(m_mpv, "idle-active").toBool())
  316. DisplayComponent::Get().restorePreviousVideoMode();
  317. }
  318. ///////////////////////////////////////////////////////////////////////////////////////////////////
  319. void PlayerComponent::onRefreshRateChange()
  320. {
  321. // Make sure settings dependent on the display refresh rate are updated properly.
  322. updateVideoSettings();
  323. }
  324. ///////////////////////////////////////////////////////////////////////////////////////////////////
  325. void PlayerComponent::updatePlaybackState()
  326. {
  327. State newState = m_state;
  328. if (m_inPlayback) {
  329. if (m_paused)
  330. {
  331. newState = State::paused;
  332. }
  333. else if (m_playbackActive)
  334. {
  335. newState = State::playing;
  336. }
  337. else
  338. {
  339. // Playback not active, but also not buffering means we're in some "other"
  340. // waiting state. Pretend to web-client that we're buffering.
  341. if (m_bufferingPercentage == 100)
  342. m_bufferingPercentage = 0;
  343. newState = State::buffering;
  344. }
  345. }
  346. else
  347. {
  348. if (!m_playbackError.isEmpty())
  349. newState = State::error;
  350. else if (m_playbackCanceled)
  351. newState = State::canceled;
  352. else
  353. newState = State::finished;
  354. }
  355. if (newState != m_state)
  356. {
  357. switch (newState) {
  358. case State::paused:
  359. QLOG_INFO() << "Entering state: paused";
  360. emit paused();
  361. break;
  362. case State::playing:
  363. QLOG_INFO() << "Entering state: playing";
  364. emit playing();
  365. break;
  366. case State::buffering:
  367. QLOG_INFO() << "Entering state: buffering";
  368. m_lastBufferingPercentage = -1; /* force update below */
  369. break;
  370. case State::finished:
  371. QLOG_INFO() << "Entering state: finished";
  372. emit finished();
  373. emit stopped();
  374. break;
  375. case State::canceled:
  376. QLOG_INFO() << "Entering state: canceled";
  377. emit canceled();
  378. emit stopped();
  379. break;
  380. case State::error:
  381. QLOG_INFO() << ("Entering state: error (" + m_playbackError + ")");
  382. emit error(m_playbackError);
  383. break;
  384. }
  385. emit stateChanged(newState, m_state);
  386. m_state = newState;
  387. }
  388. if (m_state == State::buffering && m_lastBufferingPercentage != m_bufferingPercentage)
  389. emit buffering(m_bufferingPercentage);
  390. m_lastBufferingPercentage = m_bufferingPercentage;
  391. bool is_videoPlaybackActive = m_state == State::playing && m_windowVisible;
  392. if (m_videoPlaybackActive != is_videoPlaybackActive)
  393. {
  394. m_videoPlaybackActive = is_videoPlaybackActive;
  395. emit videoPlaybackActive(m_videoPlaybackActive);
  396. }
  397. }
  398. ///////////////////////////////////////////////////////////////////////////////////////////////////
  399. void PlayerComponent::handleMpvEvent(mpv_event *event)
  400. {
  401. switch (event->event_id)
  402. {
  403. case MPV_EVENT_START_FILE:
  404. {
  405. m_inPlayback = true;
  406. break;
  407. }
  408. case MPV_EVENT_END_FILE:
  409. {
  410. mpv_event_end_file *endFile = (mpv_event_end_file *)event->data;
  411. m_inPlayback = false;
  412. m_playbackCanceled = false;
  413. m_playbackError = "";
  414. switch (endFile->reason)
  415. {
  416. case MPV_END_FILE_REASON_ERROR:
  417. {
  418. m_playbackError = mpv_error_string(endFile->error);
  419. break;
  420. }
  421. case MPV_END_FILE_REASON_STOP:
  422. {
  423. m_playbackCanceled = true;
  424. break;
  425. }
  426. }
  427. if (!m_streamSwitchImminent)
  428. m_restoreDisplayTimer.start(0);
  429. m_streamSwitchImminent = false;
  430. break;
  431. }
  432. case MPV_EVENT_PROPERTY_CHANGE:
  433. {
  434. mpv_event_property *prop = (mpv_event_property *)event->data;
  435. if (strcmp(prop->name, "pause") == 0 && prop->format == MPV_FORMAT_FLAG)
  436. {
  437. m_paused = !!*(int *)prop->data;
  438. }
  439. else if (strcmp(prop->name, "core-idle") == 0 && prop->format == MPV_FORMAT_FLAG)
  440. {
  441. m_playbackActive = !*(int *)prop->data;
  442. }
  443. else if (strcmp(prop->name, "cache-buffering-state") == 0)
  444. {
  445. m_bufferingPercentage = prop->format == MPV_FORMAT_INT64 ? *(int64_t *)prop->data : 100;
  446. }
  447. else if (strcmp(prop->name, "playback-time") == 0 && prop->format == MPV_FORMAT_DOUBLE)
  448. {
  449. double pos = *(double*)prop->data;
  450. if (fabs(pos - m_lastPositionUpdate) > 0.015)
  451. {
  452. quint64 ms = (quint64)(qMax(pos * 1000.0, 0.0));
  453. emit positionUpdate(ms);
  454. m_lastPositionUpdate = pos;
  455. }
  456. }
  457. else if (strcmp(prop->name, "vo-configured") == 0)
  458. {
  459. int state = prop->format == MPV_FORMAT_FLAG ? *(int *)prop->data : 0;
  460. m_windowVisible = state;
  461. emit windowVisible(m_windowVisible);
  462. }
  463. else if (strcmp(prop->name, "duration") == 0)
  464. {
  465. if (prop->format == MPV_FORMAT_DOUBLE)
  466. emit updateDuration(*(double *)prop->data * 1000.0);
  467. }
  468. else if (strcmp(prop->name, "audio-device-list") == 0)
  469. {
  470. updateAudioDeviceList();
  471. }
  472. else if (strcmp(prop->name, "video-dec-params") == 0)
  473. {
  474. // Aspect might be known now (or it changed during playback), so update settings
  475. // dependent on the aspect ratio.
  476. updateVideoAspectSettings();
  477. }
  478. break;
  479. }
  480. case MPV_EVENT_LOG_MESSAGE:
  481. {
  482. mpv_event_log_message *msg = (mpv_event_log_message *)event->data;
  483. // Strip the trailing '\n'
  484. size_t len = strlen(msg->text);
  485. if (len > 0 && msg->text[len - 1] == '\n')
  486. len -= 1;
  487. QString logline = QString::fromUtf8(msg->prefix) + ": " + QString::fromUtf8(msg->text, (int)len);
  488. if (msg->log_level >= MPV_LOG_LEVEL_V)
  489. QLOG_DEBUG() << qPrintable(logline);
  490. else if (msg->log_level >= MPV_LOG_LEVEL_INFO)
  491. QLOG_INFO() << qPrintable(logline);
  492. else if (msg->log_level >= MPV_LOG_LEVEL_WARN)
  493. QLOG_WARN() << qPrintable(logline);
  494. else
  495. QLOG_ERROR() << qPrintable(logline);
  496. break;
  497. }
  498. case MPV_EVENT_CLIENT_MESSAGE:
  499. {
  500. mpv_event_client_message *msg = (mpv_event_client_message *)event->data;
  501. if (msg->num_args < 3 || strcmp(msg->args[0], "hook_run") != 0)
  502. break;
  503. QString resumeId = QString::fromUtf8(msg->args[2]);
  504. // Start "on_load" hook.
  505. // This happens when the player is about to load the file, but no actual loading has taken part yet.
  506. // We use this to block loading until we explicitly tell it to continue.
  507. if (!strcmp(msg->args[1], "1"))
  508. {
  509. // Calling this lambda will instruct mpv to continue loading the file.
  510. auto resume = [=] {
  511. QLOG_INFO() << "checking codecs";
  512. startCodecsLoading([=] {
  513. QLOG_INFO() << "resuming loading";
  514. mpv::qt::command(m_mpv, QStringList() << "hook-ack" << resumeId);
  515. });
  516. };
  517. if (switchDisplayFrameRate())
  518. {
  519. // Now wait for some time for mode change - this is needed because mode changing can take some
  520. // time, during which the screen is black, and initializing hardware decoding could fail due
  521. // to various strange OS-related reasons.
  522. // (Better hope the user doesn't try to exit Konvergo during mode change.)
  523. int pause = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "refreshrate.delay").toInt() * 1000;
  524. QLOG_INFO() << "waiting" << pause << "msec after rate switch before loading";
  525. QTimer::singleShot(pause, resume);
  526. }
  527. else
  528. {
  529. resume();
  530. }
  531. break;
  532. }
  533. // Start "on_preloaded" hook.
  534. // Used initialize stream selections and to probe codecs.
  535. if (!strcmp(msg->args[1], "2"))
  536. {
  537. reselectStream(m_currentSubtitleStream, MediaType::Subtitle);
  538. reselectStream(m_currentAudioStream, MediaType::Audio);
  539. startCodecsLoading([=] {
  540. mpv::qt::command(m_mpv, QStringList() << "hook-ack" << resumeId);
  541. });
  542. break;
  543. }
  544. break;
  545. }
  546. default:; /* ignore */
  547. }
  548. }
  549. ///////////////////////////////////////////////////////////////////////////////////////////////////
  550. void PlayerComponent::handleMpvEvents()
  551. {
  552. // Process all events, until the event queue is empty.
  553. while (1)
  554. {
  555. mpv_event *event = mpv_wait_event(m_mpv, 0);
  556. if (event->event_id == MPV_EVENT_NONE)
  557. break;
  558. handleMpvEvent(event);
  559. }
  560. // Once we got all status updates, determine the new canonical state.
  561. updatePlaybackState();
  562. }
  563. ///////////////////////////////////////////////////////////////////////////////////////////////////
  564. void PlayerComponent::setVideoOnlyMode(bool enable)
  565. {
  566. if (m_window)
  567. {
  568. QQuickItem *web = m_window->findChild<QQuickItem *>("web");
  569. if (web)
  570. web->setVisible(!enable);
  571. }
  572. }
  573. ///////////////////////////////////////////////////////////////////////////////////////////////////
  574. void PlayerComponent::play()
  575. {
  576. QStringList args = (QStringList() << "set" << "pause" << "no");
  577. mpv::qt::command(m_mpv, args);
  578. }
  579. ///////////////////////////////////////////////////////////////////////////////////////////////////
  580. void PlayerComponent::stop()
  581. {
  582. QStringList args("stop");
  583. mpv::qt::command(m_mpv, args);
  584. }
  585. ///////////////////////////////////////////////////////////////////////////////////////////////////
  586. void PlayerComponent::clearQueue()
  587. {
  588. QStringList args("playlist_clear");
  589. mpv::qt::command(m_mpv, args);
  590. }
  591. ///////////////////////////////////////////////////////////////////////////////////////////////////
  592. void PlayerComponent::pause()
  593. {
  594. QStringList args = (QStringList() << "set" << "pause" << "yes");
  595. mpv::qt::command(m_mpv, args);
  596. }
  597. ///////////////////////////////////////////////////////////////////////////////////////////////////
  598. void PlayerComponent::seekTo(qint64 ms)
  599. {
  600. double timeSecs = ms / 1000.0;
  601. QVariantList args = (QVariantList() << "seek" << timeSecs << "absolute+exact");
  602. mpv::qt::command(m_mpv, args);
  603. }
  604. ///////////////////////////////////////////////////////////////////////////////////////////////////
  605. QVariant PlayerComponent::getAudioDeviceList()
  606. {
  607. return mpv::qt::get_property(m_mpv, "audio-device-list");
  608. }
  609. ///////////////////////////////////////////////////////////////////////////////////////////////////
  610. void PlayerComponent::setAudioDevice(const QString& name)
  611. {
  612. mpv::qt::set_property(m_mpv, "audio-device", name);
  613. }
  614. ///////////////////////////////////////////////////////////////////////////////////////////////////
  615. void PlayerComponent::setVolume(int volume)
  616. {
  617. // Will fail if no audio output opened (i.e. no file playing)
  618. mpv::qt::set_property(m_mpv, "volume", volume);
  619. }
  620. ///////////////////////////////////////////////////////////////////////////////////////////////////
  621. int PlayerComponent::volume()
  622. {
  623. QVariant volume = mpv::qt::get_property(m_mpv, "volume");
  624. if (volume.isValid())
  625. return volume.toInt();
  626. return 0;
  627. }
  628. ///////////////////////////////////////////////////////////////////////////////////////////////////
  629. void PlayerComponent::setMuted(bool muted)
  630. {
  631. // Will fail if no audio output opened (i.e. no file playing)
  632. mpv::qt::set_property(m_mpv, "mute", muted);
  633. }
  634. ///////////////////////////////////////////////////////////////////////////////////////////////////
  635. bool PlayerComponent::muted()
  636. {
  637. QVariant mute = mpv::qt::get_property(m_mpv, "mute");
  638. if (mute.isValid())
  639. return mute.toBool();
  640. return false;
  641. }
  642. ///////////////////////////////////////////////////////////////////////////////////////////////////
  643. QVariantList PlayerComponent::findStreamsForURL(const QString &url)
  644. {
  645. bool isExternal = !url.isEmpty();
  646. QVariantList res;
  647. auto tracks = mpv::qt::get_property(m_mpv, "track-list");
  648. for (auto track : tracks.toList())
  649. {
  650. QVariantMap map = track.toMap();
  651. if (map["external"].toBool() != isExternal)
  652. continue;
  653. if (!isExternal || map["external-filename"].toString() == url)
  654. res += map;
  655. }
  656. return res;
  657. }
  658. ///////////////////////////////////////////////////////////////////////////////////////////////////
  659. void PlayerComponent::reselectStream(const QString &streamSelection, MediaType target)
  660. {
  661. QString streamIdPropertyName;
  662. QString streamAddCommandName;
  663. QString mpvStreamTypeName;
  664. switch (target)
  665. {
  666. case MediaType::Subtitle:
  667. mpvStreamTypeName = "sub";
  668. streamIdPropertyName = "sid";
  669. streamAddCommandName = "sub-add";
  670. break;
  671. case MediaType::Audio:
  672. mpvStreamTypeName = "audio";
  673. streamIdPropertyName = "aid";
  674. streamAddCommandName = "audio-add";
  675. break;
  676. }
  677. QString streamName;
  678. QString streamID;
  679. if (streamSelection.startsWith("#"))
  680. {
  681. int splitPos = streamSelection.indexOf(",");
  682. if (splitPos < 0)
  683. {
  684. // Stream from the main file
  685. streamID = streamSelection.mid(1);
  686. streamName = "";
  687. }
  688. else
  689. {
  690. // Stream from an external file
  691. streamID = streamSelection.mid(1, splitPos - 1);
  692. streamName = streamSelection.mid(splitPos + 1);
  693. }
  694. }
  695. else if (!streamSelection.isEmpty())
  696. {
  697. if (target == MediaType::Audio)
  698. {
  699. // For some reason, audio stream selections never start with '#'.
  700. streamID = streamSelection;
  701. streamName = "";
  702. }
  703. else
  704. {
  705. // Legacy web-client single external subtitle
  706. streamID = "0";
  707. streamName = streamSelection;
  708. }
  709. }
  710. if (!streamName.isEmpty())
  711. {
  712. if (streamName.startsWith("https://")) {
  713. QUrl qurl = streamName;
  714. qurl.setHost(ConvertPlexDirectURL(qurl.host()));
  715. streamName = qurl.toString();
  716. }
  717. auto streams = findStreamsForURL(streamName);
  718. if (streams.isEmpty())
  719. {
  720. QStringList args = (QStringList() << streamAddCommandName << streamName);
  721. mpv::qt::command(m_mpv, args);
  722. }
  723. }
  724. QString selection = "no";
  725. for (auto stream : findStreamsForURL(streamName))
  726. {
  727. auto map = stream.toMap();
  728. if (map["type"].toString() != mpvStreamTypeName)
  729. continue;
  730. if (!streamID.isEmpty() && map["ff-index"].toString() == streamID)
  731. {
  732. selection = map["id"].toString();
  733. break;
  734. }
  735. }
  736. // Fallback to the first stream if none could be found.
  737. // Useful if web-client uses wrong stream IDs when e.g. transcoding.
  738. if ((target == MediaType::Audio || !streamID.isEmpty()) && selection == "no")
  739. selection = "1";
  740. mpv::qt::set_property(m_mpv, streamIdPropertyName, selection);
  741. }
  742. ///////////////////////////////////////////////////////////////////////////////////////////////////
  743. void PlayerComponent::setSubtitleStream(const QString &subtitleStream)
  744. {
  745. m_currentSubtitleStream = subtitleStream;
  746. reselectStream(m_currentSubtitleStream, MediaType::Subtitle);
  747. }
  748. ///////////////////////////////////////////////////////////////////////////////////////////////////
  749. void PlayerComponent::setAudioStream(const QString &audioStream)
  750. {
  751. m_currentAudioStream = audioStream;
  752. reselectStream(m_currentAudioStream, MediaType::Audio);
  753. }
  754. /////////////////////////////////////////////////////////////////////////////////////////
  755. void PlayerComponent::setAudioDelay(qint64 milliseconds)
  756. {
  757. m_playbackAudioDelay = milliseconds;
  758. double displayFps = DisplayComponent::Get().currentRefreshRate();
  759. const char *audioDelaySetting = "audio_delay.normal";
  760. if (fabs(displayFps - 24) < 0.5) // cover 24Hz, 23.976Hz, and values very close
  761. audioDelaySetting = "audio_delay.24hz";
  762. else if (fabs(displayFps - 25) < 0.5)
  763. audioDelaySetting = "audio_delay.25hz";
  764. else if (fabs(displayFps - 50) < 0.5)
  765. audioDelaySetting = "audio_delay.50hz";
  766. double fixedDelay = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, audioDelaySetting).toFloat();
  767. mpv::qt::set_property(m_mpv, "audio-delay", (fixedDelay + m_playbackAudioDelay) / 1000.0);
  768. }
  769. /////////////////////////////////////////////////////////////////////////////////////////
  770. void PlayerComponent::setSubtitleDelay(qint64 milliseconds)
  771. {
  772. mpv::qt::set_property(m_mpv, "sub-delay", milliseconds / 1000.0);
  773. }
  774. ///////////////////////////////////////////////////////////////////////////////////////////////////
  775. // This is called with the set of previous audio devices that were detected, and the set of current
  776. // audio devices. From this we guess whether we should reopen the audio device. If the user-selected
  777. // device went away previously, and now comes back, reinitializing the player's audio output will
  778. // force the player and/or the OS to move audio output back to the user-selected device.
  779. void PlayerComponent::checkCurrentAudioDevice(const QSet<QString>& old_devs, const QSet<QString>& new_devs)
  780. {
  781. QString userDevice = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device").toString();
  782. QSet<QString> removed = old_devs - new_devs;
  783. QSet<QString> added = new_devs - old_devs;
  784. QLOG_DEBUG() << "Audio devices removed:" << removed;
  785. QLOG_DEBUG() << "Audio devices added:" << added;
  786. QLOG_DEBUG() << "Audio device selected:" << userDevice;
  787. if (userDevice.length())
  788. {
  789. if (added.contains(userDevice) || removed.contains(userDevice))
  790. {
  791. // The timer is for debouncing the reload. Several change notifications could
  792. // come in quick succession. Also, it's possible that trying to open the
  793. // reappeared audio device immediately can fail.
  794. m_reloadAudioTimer.start(500);
  795. }
  796. }
  797. }
  798. ///////////////////////////////////////////////////////////////////////////////////////////////////
  799. void PlayerComponent::updateAudioDeviceList()
  800. {
  801. QString userDevice = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device").toString();
  802. bool userDeviceFound = false;
  803. QVariantList settingList;
  804. QVariant list = getAudioDeviceList();
  805. QSet<QString> devices;
  806. for(const QVariant& d : list.toList())
  807. {
  808. Q_ASSERT(d.type() == QVariant::Map);
  809. QVariantMap dmap = d.toMap();
  810. QString device = dmap["name"].toString();
  811. QString description = dmap["description"].toString();
  812. devices.insert(device);
  813. if (userDevice == device)
  814. userDeviceFound = true;
  815. QVariantMap entry;
  816. entry["value"] = device;
  817. entry["title"] = description;
  818. settingList << entry;
  819. }
  820. if (!userDeviceFound)
  821. {
  822. QVariantMap entry;
  823. entry["value"] = userDevice;
  824. entry["title"] = "[Disconnected device: " + userDevice + "]";
  825. settingList << entry;
  826. }
  827. SettingsComponent::Get().updatePossibleValues(SETTINGS_SECTION_AUDIO, "device", settingList);
  828. checkCurrentAudioDevice(m_audioDevices, devices);
  829. m_audioDevices = devices;
  830. }
  831. ///////////////////////////////////////////////////////////////////////////////////////////////////
  832. void PlayerComponent::updateAudioDevice()
  833. {
  834. QString device = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device").toString();
  835. if (!m_audioDevices.contains(device))
  836. {
  837. QLOG_WARN() << "Not using audio device" << device << "because it's not present.";
  838. device = "auto";
  839. }
  840. mpv::qt::set_property(m_mpv, "audio-device", device);
  841. }
  842. ///////////////////////////////////////////////////////////////////////////////////////////////////
  843. void PlayerComponent::setAudioConfiguration()
  844. {
  845. QString deviceType = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "devicetype").toString();
  846. mpv::qt::set_property(m_mpv, "audio-exclusive", SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "exclusive").toBool());
  847. updateAudioDevice();
  848. QString resampleOpts = "";
  849. bool normalize = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "normalize").toBool();
  850. resampleOpts += QString(":normalize=") + (normalize ? "yes" : "no");
  851. // Make downmix more similar to PHT.
  852. resampleOpts += ":o=[surround_mix_level=1]";
  853. mpv::qt::set_property(m_mpv, "af-defaults", "lavrresample" + resampleOpts);
  854. m_passthroughCodecs.clear();
  855. // passthrough doesn't make sense with basic type
  856. if (deviceType != AUDIO_DEVICE_TYPE_BASIC)
  857. {
  858. SettingsSection* audioSection = SettingsComponent::Get().getSection(SETTINGS_SECTION_AUDIO);
  859. QStringList codecs;
  860. if (deviceType == AUDIO_DEVICE_TYPE_SPDIF)
  861. codecs = AudioCodecsSPDIF();
  862. else if (deviceType == AUDIO_DEVICE_TYPE_HDMI)
  863. codecs = AudioCodecsAll();
  864. for(const QString& key : codecs)
  865. {
  866. if (audioSection->value("passthrough." + key).toBool())
  867. m_passthroughCodecs << key;
  868. }
  869. // dts-hd includes dts, but listing dts before dts-hd may disable dts-hd.
  870. if (m_passthroughCodecs.indexOf("dts-hd") != -1)
  871. m_passthroughCodecs.removeAll("dts");
  872. }
  873. QString passthroughCodecs = m_passthroughCodecs.join(",");
  874. mpv::qt::set_property(m_mpv, "audio-spdif", passthroughCodecs);
  875. // set the channel layout
  876. QVariant layout = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "channels");
  877. // always force either stereo or transcoding
  878. if (deviceType == AUDIO_DEVICE_TYPE_SPDIF)
  879. layout = "2.0";
  880. mpv::qt::set_property(m_mpv, "audio-channels", layout);
  881. // if the user has indicated that PCM only works for stereo, and that
  882. // the receiver supports AC3, set this extra option that allows us to transcode
  883. // 5.1 audio into a usable format, note that we only support AC3
  884. // here for now. We might need to add support for DTS transcoding
  885. // if we see user requests for it.
  886. //
  887. m_doAc3Transcoding = false;
  888. if (deviceType == AUDIO_DEVICE_TYPE_SPDIF &&
  889. SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "passthrough.ac3").toBool())
  890. {
  891. QString filterArgs = "";
  892. mpv::qt::command(m_mpv, QStringList() << "af" << "add" << ("@ac3:lavcac3enc" + filterArgs));
  893. m_doAc3Transcoding = true;
  894. }
  895. else
  896. {
  897. mpv::qt::command(m_mpv, QStringList() << "af" << "del" << "@ac3");
  898. }
  899. QVariant device = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device");
  900. // Make a informational log message.
  901. QString audioConfig = QString(QString("Audio Config - device: %1, ") +
  902. "channel layout: %2, " +
  903. "passthrough codecs: %3, " +
  904. "ac3 transcoding: %4").arg(device.toString(),
  905. layout.toString(),
  906. passthroughCodecs.isEmpty() ? "none" : passthroughCodecs,
  907. m_doAc3Transcoding ? "yes" : "no");
  908. QLOG_INFO() << qPrintable(audioConfig);
  909. }
  910. ///////////////////////////////////////////////////////////////////////////////////////////////////
  911. void PlayerComponent::updateSubtitleSettings()
  912. {
  913. QVariant size = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "size");
  914. mpv::qt::set_property(m_mpv, "sub-text-font-size", size);
  915. QVariant colorsString = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "color");
  916. auto colors = colorsString.toString().split(",");
  917. if (colors.length() == 2)
  918. {
  919. mpv::qt::set_property(m_mpv, "sub-text-color", colors[0]);
  920. mpv::qt::set_property(m_mpv, "sub-text-border-color", colors[1]);
  921. }
  922. QVariant subposString = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "placement");
  923. auto subpos = subposString.toString().split(",");
  924. if (subpos.length() == 2)
  925. {
  926. mpv::qt::set_property(m_mpv, "sub-text-align-x", subpos[0]);
  927. mpv::qt::set_property(m_mpv, "sub-text-align-y", subpos[1]);
  928. }
  929. }
  930. ///////////////////////////////////////////////////////////////////////////////////////////////////
  931. void PlayerComponent::updateVideoAspectSettings()
  932. {
  933. QVariant mode = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "aspect").toString();
  934. bool disableScaling = false;
  935. bool keepAspect = true;
  936. QString forceAspect = "-1";
  937. double panScan = 0.0;
  938. if (mode == "custom")
  939. {
  940. // in particular, do not restore anything - the intention is not to touch the user's mpv.conf settings, or whatever
  941. return;
  942. }
  943. else if (mode == "zoom")
  944. {
  945. panScan = 1.0;
  946. }
  947. else if (mode == "force_4_3")
  948. {
  949. forceAspect = "4:3";
  950. }
  951. else if (mode == "force_16_9")
  952. {
  953. forceAspect = "16:9";
  954. }
  955. else if (mode == "force_16_9_if_4_3")
  956. {
  957. auto params = mpv::qt::get_property(m_mpv, "video-dec-params").toMap();
  958. auto aspect = params["aspect"].toFloat();
  959. if (fabs(aspect - 4.0/3.0) < 0.1)
  960. forceAspect = "16:9";
  961. }
  962. else if (mode == "stretch")
  963. {
  964. keepAspect = false;
  965. }
  966. else if (mode == "noscaling")
  967. {
  968. disableScaling = true;
  969. }
  970. mpv::qt::set_property(m_mpv, "video-unscaled", disableScaling);
  971. mpv::qt::set_property(m_mpv, "video-aspect", forceAspect);
  972. mpv::qt::set_property(m_mpv, "keepaspect", keepAspect);
  973. mpv::qt::set_property(m_mpv, "panscan", panScan);
  974. }
  975. ///////////////////////////////////////////////////////////////////////////////////////////////////
  976. void PlayerComponent::updateVideoSettings()
  977. {
  978. if (!m_mpv)
  979. return;
  980. QVariant syncMode = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "sync_mode");
  981. mpv::qt::set_property(m_mpv, "video-sync", syncMode);
  982. QString hardwareDecodingMode = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "hardwareDecoding").toString();
  983. QString hwdecMode = "no";
  984. QString hwdecVTFormat = "nv12";
  985. if (hardwareDecodingMode == "enabled")
  986. hwdecMode = "auto";
  987. else if (hardwareDecodingMode == "osx_compat")
  988. {
  989. hwdecMode = "auto";
  990. hwdecVTFormat = "uyvy422";
  991. }
  992. else if (hardwareDecodingMode == "copy")
  993. {
  994. hwdecMode = "auto-copy";
  995. }
  996. mpv::qt::set_property(m_mpv, "hwdec", hwdecMode);
  997. mpv::qt::set_property(m_mpv, "videotoolbox-format", hwdecVTFormat);
  998. QVariant deinterlace = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "deinterlace");
  999. mpv::qt::set_property(m_mpv, "deinterlace", deinterlace.toBool() ? "yes" : "no");
  1000. #ifndef TARGET_RPI
  1001. double displayFps = DisplayComponent::Get().currentRefreshRate();
  1002. mpv::qt::set_property(m_mpv, "display-fps", displayFps);
  1003. #endif
  1004. setAudioDelay(m_playbackAudioDelay);
  1005. QVariant cache = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "cache");
  1006. mpv::qt::set_property(m_mpv, "cache", cache.toInt() * 1024);
  1007. updateVideoAspectSettings();
  1008. }
  1009. /////////////////////////////////////////////////////////////////////////////////////////
  1010. void PlayerComponent::userCommand(QString command)
  1011. {
  1012. QByteArray cmdUtf8 = command.toUtf8();
  1013. mpv_command_string(m_mpv, cmdUtf8.data());
  1014. }
  1015. /////////////////////////////////////////////////////////////////////////////////////////
  1016. void PlayerComponent::initializeCodecSupport()
  1017. {
  1018. QMap<QString, QString> all = { {"vc1", "WVC1"}, {"mpeg2video", "MPG2"} };
  1019. for (auto name : all.keys())
  1020. {
  1021. bool ok = true;
  1022. #ifdef TARGET_RPI
  1023. char res[100] = "";
  1024. bcm_host_init();
  1025. if (vc_gencmd(res, sizeof(res), "codec_enabled %s", all[name].toUtf8().data()))
  1026. res[0] = '\0'; // error
  1027. ok = !!strstr(res, "=enabled");
  1028. #endif
  1029. m_codecSupport[name] = ok;
  1030. QLOG_INFO() << "Codec" << name << (ok ? "present" : "disabled");
  1031. }
  1032. }
  1033. /////////////////////////////////////////////////////////////////////////////////////////
  1034. bool PlayerComponent::checkCodecSupport(const QString& codec)
  1035. {
  1036. if (m_codecSupport.contains(codec))
  1037. return m_codecSupport[codec];
  1038. return true; // doesn't matter if unknown codecs are reported as "ok"
  1039. }
  1040. /////////////////////////////////////////////////////////////////////////////////////////
  1041. QList<CodecDriver> convertCodecList(QVariant list, CodecType type)
  1042. {
  1043. QList<CodecDriver> codecs;
  1044. foreach (const QVariant& e, list.toList())
  1045. {
  1046. QVariantMap map = e.toMap();
  1047. QString family = map["family"].toString();
  1048. QString codec = map["codec"].toString();
  1049. QString driver = map["driver"].toString();
  1050. // Only include FFmpeg codecs; exclude pseudo-codecs like spdif.
  1051. if (family != "lavc")
  1052. continue;
  1053. CodecDriver ncodec = {};
  1054. ncodec.type = type;
  1055. ncodec.format = codec;
  1056. ncodec.driver = driver;
  1057. ncodec.present = true;
  1058. codecs.append(ncodec);
  1059. }
  1060. return codecs;
  1061. }
  1062. /////////////////////////////////////////////////////////////////////////////////////////
  1063. QList<CodecDriver> PlayerComponent::installedCodecDrivers()
  1064. {
  1065. QList<CodecDriver> codecs;
  1066. codecs.append(convertCodecList(mpv::qt::get_property(m_mpv, "decoder-list"), CodecType::Decoder));
  1067. codecs.append(convertCodecList(mpv::qt::get_property(m_mpv, "encoder-list"), CodecType::Encoder));
  1068. return codecs;
  1069. }
  1070. /////////////////////////////////////////////////////////////////////////////////////////
  1071. QStringList PlayerComponent::installedDecoderCodecs()
  1072. {
  1073. QStringList formats;
  1074. bool hasPcm = false;
  1075. for (auto driver : installedCodecDrivers())
  1076. {
  1077. if (driver.type == CodecType::Decoder && checkCodecSupport(driver.format))
  1078. {
  1079. QString name = Codecs::plexNameFromFF(driver.format);
  1080. if (name.startsWith("pcm_") && name != "pcm_bluray" && name != "pcm_dvd")
  1081. {
  1082. if (hasPcm)
  1083. continue;
  1084. hasPcm = true;
  1085. name = "pcm";
  1086. }
  1087. formats.append(name);
  1088. }
  1089. }
  1090. return formats;
  1091. }
  1092. /////////////////////////////////////////////////////////////////////////////////////////
  1093. PlaybackInfo PlayerComponent::getPlaybackInfo()
  1094. {
  1095. PlaybackInfo info = {};
  1096. for (auto codec : m_passthroughCodecs)
  1097. {
  1098. // Normalize back to canonical codec names.
  1099. if (codec == "dts-hd")
  1100. codec = "dts";
  1101. info.audioPassthroughCodecs.insert(codec);
  1102. }
  1103. info.enableAC3Transcoding = m_doAc3Transcoding;
  1104. auto tracks = mpv::qt::get_property(m_mpv, "track-list");
  1105. for (auto track : tracks.toList())
  1106. {
  1107. QVariantMap map = track.toMap();
  1108. QString type = map["type"].toString();
  1109. StreamInfo stream = {};
  1110. stream.isVideo = type == "video";
  1111. stream.isAudio = type == "audio";
  1112. stream.codec = map["codec"].toString();
  1113. stream.audioChannels = map["demux-channel-count"].toInt();
  1114. stream.audioSampleRate = map["demux-samplerate"].toInt();
  1115. stream.videoResolution = QSize(map["demux-w"].toInt(), map["demux-h"].toInt());
  1116. // Get the profile from the server, because mpv can't determine it yet.
  1117. if (stream.isVideo)
  1118. {
  1119. int index = map["ff-index"].toInt();
  1120. for (auto partInfo : m_serverMediaInfo["Part"].toList())
  1121. {
  1122. for (auto streamInfo : partInfo.toMap()["Stream"].toList())
  1123. {
  1124. auto streamInfoMap = streamInfo.toMap();
  1125. bool ok = false;
  1126. if (streamInfoMap["index"].toInt(&ok) == index && ok)
  1127. {
  1128. stream.profile = streamInfoMap["profile"].toString();
  1129. QLOG_DEBUG() << "h264profile:" << stream.profile;
  1130. }
  1131. }
  1132. }
  1133. }
  1134. info.streams.append(stream);
  1135. }
  1136. // If we're in an early stage where we don't have streams yet, try to get the
  1137. // info from the PMS metadata.
  1138. if (!info.streams.size())
  1139. {
  1140. for (auto partInfo : m_serverMediaInfo["Part"].toList())
  1141. {
  1142. for (auto streamInfo : partInfo.toMap()["Stream"].toList())
  1143. {
  1144. auto streamInfoMap = streamInfo.toMap();
  1145. StreamInfo stream = {};
  1146. stream.isVideo = streamInfoMap["width"].isValid();
  1147. stream.isAudio = streamInfoMap["channels"].isValid();
  1148. stream.codec = Codecs::plexNameToFF(streamInfoMap["codec"].toString());
  1149. stream.audioChannels = streamInfoMap["channels"].toInt();
  1150. stream.videoResolution = QSize(streamInfoMap["width"].toInt(), streamInfoMap["height"].toInt());
  1151. stream.profile = streamInfoMap["profile"].toString();
  1152. info.streams.append(stream);
  1153. }
  1154. }
  1155. }
  1156. return info;
  1157. }
  1158. /////////////////////////////////////////////////////////////////////////////////////////
  1159. void PlayerComponent::setPreferredCodecs(const QList<CodecDriver>& codecs)
  1160. {
  1161. QStringList items;
  1162. for (auto codec : codecs)
  1163. {
  1164. if (codec.type == CodecType::Decoder)
  1165. {
  1166. items << codec.driver;
  1167. }
  1168. }
  1169. QString opt = items.join(",");
  1170. // For simplicity, we don't distinguish between audio and video. The player
  1171. // will ignore entries with mismatching media type.
  1172. mpv::qt::set_property(m_mpv, "ad", opt);
  1173. mpv::qt::set_property(m_mpv, "vd", opt);
  1174. }
  1175. // For QVariant.
  1176. Q_DECLARE_METATYPE(std::function<void()>);
  1177. /////////////////////////////////////////////////////////////////////////////////////////
  1178. void PlayerComponent::startCodecsLoading(std::function<void()> resume)
  1179. {
  1180. auto fetcher = new CodecsFetcher();
  1181. fetcher->userData = QVariant::fromValue(resume);
  1182. connect(fetcher, &CodecsFetcher::done, this, &PlayerComponent::onCodecsLoadingDone);
  1183. Codecs::updateCachedCodecList();
  1184. QList<CodecDriver> codecs = Codecs::determineRequiredCodecs(getPlaybackInfo());
  1185. setPreferredCodecs(codecs);
  1186. fetcher->installCodecs(codecs);
  1187. }
  1188. /////////////////////////////////////////////////////////////////////////////////////////
  1189. void PlayerComponent::onCodecsLoadingDone(CodecsFetcher* sender)
  1190. {
  1191. sender->deleteLater();
  1192. sender->userData.value<std::function<void()>>()();
  1193. }
  1194. /////////////////////////////////////////////////////////////////////////////////////////
  1195. static QString get_mpv_osd(mpv_handle *ctx, const QString& property)
  1196. {
  1197. char *s = mpv_get_property_osd_string(ctx, property.toUtf8().data());
  1198. if (!s)
  1199. return "-";
  1200. QString r = QString::fromUtf8(s);
  1201. mpv_free(s);
  1202. if (r.size() > 400)
  1203. r = r.mid(0, 400) + "...";
  1204. Log::CensorAuthTokens(r);
  1205. return r;
  1206. }
  1207. #define MPV_PROPERTY(p) get_mpv_osd(m_mpv, p)
  1208. #define MPV_PROPERTY_BOOL(p) (mpv::qt::get_property(m_mpv, p).toBool())
  1209. /////////////////////////////////////////////////////////////////////////////////////////
  1210. void PlayerComponent::appendAudioFormat(QTextStream& info, const QString& property) const
  1211. {
  1212. // Guess if it's a passthrough format. Don't show the channel layout in this
  1213. // case, because it's confusing.
  1214. QString audioFormat = MPV_PROPERTY(property + "/format");
  1215. if (audioFormat.startsWith("spdif-"))
  1216. {
  1217. info << "passthrough (" << audioFormat.mid(6) << ")";
  1218. }
  1219. else
  1220. {
  1221. QString hr = MPV_PROPERTY(property + "/hr-channels");
  1222. QString full = MPV_PROPERTY(property + "/channels");
  1223. info << hr;
  1224. if (hr != full)
  1225. info << " (" << full << ")";
  1226. }
  1227. }
  1228. /////////////////////////////////////////////////////////////////////////////////////////
  1229. QString PlayerComponent::videoInformation() const
  1230. {
  1231. QString infoStr;
  1232. QTextStream info(&infoStr);
  1233. // check if video is playing
  1234. if (mpv::qt::get_property(m_mpv, "idle-active").toBool())
  1235. return "";
  1236. info << "File:" << endl;
  1237. info << "URL: " << MPV_PROPERTY("path") << endl;
  1238. info << "Container: " << MPV_PROPERTY("file-format") << endl;
  1239. info << "Native seeking: " << ((MPV_PROPERTY_BOOL("seekable") &&
  1240. !MPV_PROPERTY_BOOL("partially-seekable"))
  1241. ? "yes" : "no") << endl;
  1242. info << endl;
  1243. info << "Video:" << endl;
  1244. info << "Codec: " << MPV_PROPERTY("video-codec") << endl;
  1245. info << "Size: " << MPV_PROPERTY("video-params/dw") << "x"
  1246. << MPV_PROPERTY("video-params/dh") << endl;
  1247. info << "FPS (container): " << MPV_PROPERTY("container-fps") << endl;
  1248. info << "FPS (filters): " << MPV_PROPERTY("estimated-vf-fps") << endl;
  1249. info << "Aspect: " << MPV_PROPERTY("video-aspect") << endl;
  1250. info << "Bitrate: " << MPV_PROPERTY("video-bitrate") << endl;
  1251. double displayFps = DisplayComponent::Get().currentRefreshRate();
  1252. info << "Display FPS: " << MPV_PROPERTY("display-fps")
  1253. << " (" << displayFps << ")" << endl;
  1254. info << "Hardware Decoding: " << MPV_PROPERTY("hwdec-current")
  1255. << " (" << MPV_PROPERTY("hwdec-interop") << ")" << endl;
  1256. info << endl;
  1257. info << "Audio: " << endl;
  1258. info << "Codec: " << MPV_PROPERTY("audio-codec") << endl;
  1259. info << "Bitrate: " << MPV_PROPERTY("audio-bitrate") << endl;
  1260. info << "Channels: ";
  1261. appendAudioFormat(info, "audio-params");
  1262. info << " -> ";
  1263. appendAudioFormat(info, "audio-out-params");
  1264. info << endl;
  1265. info << "Output driver: " << MPV_PROPERTY("current-ao") << endl;
  1266. info << endl;
  1267. info << "Performance: " << endl;
  1268. info << "A/V: " << MPV_PROPERTY("avsync") << endl;
  1269. info << "Dropped frames: " << MPV_PROPERTY("vo-drop-frame-count") << endl;
  1270. bool dispSync = MPV_PROPERTY_BOOL("display-sync-active");
  1271. info << "Display Sync: ";
  1272. if (!dispSync)
  1273. {
  1274. info << "no" << endl;
  1275. }
  1276. else
  1277. {
  1278. info << "yes (ratio " << MPV_PROPERTY("vsync-ratio") << ")" << endl;
  1279. info << "Mistimed frames: " << MPV_PROPERTY("mistimed-frame-count")
  1280. << "/" << MPV_PROPERTY("vo-delayed-frame-count") << endl;
  1281. info << "Measured FPS: " << MPV_PROPERTY("estimated-display-fps")
  1282. << " (" << MPV_PROPERTY("vsync-jitter") << ")" << endl;
  1283. info << "V. speed corr.: " << MPV_PROPERTY("video-speed-correction") << endl;
  1284. info << "A. speed corr.: " << MPV_PROPERTY("audio-speed-correction") << endl;
  1285. }
  1286. info << endl;
  1287. info << "Cache:" << endl;
  1288. info << "Seconds: " << MPV_PROPERTY("demuxer-cache-duration") << endl;
  1289. info << "Extra readahead: " << MPV_PROPERTY("cache-used") << endl;
  1290. info << "Buffering: " << MPV_PROPERTY("cache-buffering-state") << endl;
  1291. info << "Speed: " << MPV_PROPERTY("cache-speed") << endl;
  1292. info << endl;
  1293. info << "Misc: " << endl;
  1294. info << "Time: " << MPV_PROPERTY("playback-time") << " / "
  1295. << MPV_PROPERTY("duration")
  1296. << " (" << MPV_PROPERTY("percent-pos") << "%)" << endl;
  1297. info << "State: " << (MPV_PROPERTY_BOOL("pause") ? "paused " : "")
  1298. << (MPV_PROPERTY_BOOL("paused-for-cache") ? "buffering " : "")
  1299. << (MPV_PROPERTY_BOOL("core-idle") ? "waiting " : "playing ")
  1300. << (MPV_PROPERTY_BOOL("seeking") ? "seeking " : "")
  1301. << endl;
  1302. info << flush;
  1303. return infoStr;
  1304. }