PlayerComponent.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836
  1. #include "PlayerComponent.h"
  2. #include <QString>
  3. #include <Qt>
  4. #include <QDir>
  5. #include <QCoreApplication>
  6. #include "display/DisplayComponent.h"
  7. #include "settings/SettingsComponent.h"
  8. #include "system/SystemComponent.h"
  9. #include "utils/Utils.h"
  10. #include "ComponentManager.h"
  11. #include "settings/SettingsSection.h"
  12. #include "PlayerQuickItem.h"
  13. #include "QsLog.h"
  14. #include <math.h>
  15. #include <string.h>
  16. #include <shared/Paths.h>
  17. ///////////////////////////////////////////////////////////////////////////////////////////////////
  18. static void wakeup_cb(void *context)
  19. {
  20. PlayerComponent *player = (PlayerComponent *)context;
  21. emit player->onMpvEvents();
  22. }
  23. ///////////////////////////////////////////////////////////////////////////////////////////////////
  24. PlayerComponent::PlayerComponent(QObject* parent)
  25. : ComponentBase(parent), m_lastPositionUpdate(0.0), m_playbackAudioDelay(0), m_playbackStartSent(false), m_window(0), m_mediaFrameRate(0),
  26. m_restoreDisplayTimer(this), m_reloadAudioTimer(this)
  27. {
  28. qmlRegisterType<PlayerQuickItem>("Konvergo", 1, 0, "MpvVideo"); // deprecated name
  29. qmlRegisterType<PlayerQuickItem>("Konvergo", 1, 0, "KonvergoVideo");
  30. m_restoreDisplayTimer.setSingleShot(true);
  31. connect(&m_restoreDisplayTimer, &QTimer::timeout, this, &PlayerComponent::onRestoreDisplay);
  32. connect(&DisplayComponent::Get(), &DisplayComponent::refreshRateChanged, this, &PlayerComponent::onRefreshRateChange);
  33. m_reloadAudioTimer.setSingleShot(true);
  34. connect(&m_reloadAudioTimer, &QTimer::timeout, this, &PlayerComponent::onReloadAudio);
  35. }
  36. ///////////////////////////////////////////////////////////////////////////////////////////////////
  37. PlayerComponent::~PlayerComponent()
  38. {
  39. if (m_mpv)
  40. mpv_set_wakeup_callback(m_mpv, NULL, NULL);
  41. }
  42. ///////////////////////////////////////////////////////////////////////////////////////////////////
  43. bool PlayerComponent::componentInitialize()
  44. {
  45. m_mpv = mpv::qt::Handle::FromRawHandle(mpv_create());
  46. if (!m_mpv)
  47. throw FatalException(tr("Failed to load mpv."));
  48. mpv_request_log_messages(m_mpv, "terminal-default");
  49. mpv_set_option_string(m_mpv, "msg-level", "all=v");
  50. // No mouse events
  51. mpv_set_option_string(m_mpv, "input-cursor", "no");
  52. mpv_set_option_string(m_mpv, "cursor-autohide", "no");
  53. mpv_set_option_string(m_mpv, "config", "yes");
  54. mpv_set_option_string(m_mpv, "config-dir", Paths::dataDir().toUtf8().data());
  55. // We don't need this, so avoid initializing fontconfig.
  56. mpv_set_option_string(m_mpv, "use-text-osd", "no");
  57. // Always use the system mixer.
  58. mpv_set_option_string(m_mpv, "softvol", "no");
  59. // Just discard audio output if no audio device could be opened. This gives
  60. // us better flexibility how to react to such errors (instead of just
  61. // aborting playback immediately).
  62. mpv_set_option_string(m_mpv, "audio-fallback-to-null", "yes");
  63. // Make it load the hwdec interop, so hwdec can be enabled at runtime.
  64. mpv::qt::set_option_variant(m_mpv, "hwdec-preload", "auto");
  65. // User-visible application name used by some audio APIs (at least PulseAudio).
  66. mpv_set_option_string(m_mpv, "audio-client-name", QCoreApplication::applicationName().toUtf8().data());
  67. // User-visible stream title used by some audio APIs (at least PulseAudio and wasapi).
  68. mpv_set_option_string(m_mpv, "title", QCoreApplication::applicationName().toUtf8().data());
  69. // Apply some low-memory settings on RPI, which is relatively memory-constrained.
  70. #ifdef TARGET_RPI
  71. // The backbuffer makes seeking back faster (without having to do a HTTP-level seek)
  72. mpv::qt::set_option_variant(m_mpv, "cache-backbuffer", 10 * 1024); // KB
  73. // The demuxer queue is used for the readahead, and also for dealing with badly
  74. // interlaved audio/video. Setting it too low increases sensitivity to network
  75. // issues, and could cause playback failure with "bad" files.
  76. mpv::qt::set_option_variant(m_mpv, "demuxer-max-bytes", 50 * 1024 * 1024); // bytes
  77. #endif
  78. mpv_observe_property(m_mpv, 0, "pause", MPV_FORMAT_FLAG);
  79. mpv_observe_property(m_mpv, 0, "cache-buffering-state", MPV_FORMAT_INT64);
  80. mpv_observe_property(m_mpv, 0, "playback-time", MPV_FORMAT_DOUBLE);
  81. mpv_observe_property(m_mpv, 0, "vo-configured", MPV_FORMAT_FLAG);
  82. mpv_observe_property(m_mpv, 0, "duration", MPV_FORMAT_DOUBLE);
  83. mpv_observe_property(m_mpv, 0, "audio-device-list", MPV_FORMAT_NODE);
  84. connect(this, &PlayerComponent::onMpvEvents, this, &PlayerComponent::handleMpvEvents, Qt::QueuedConnection);
  85. mpv_set_wakeup_callback(m_mpv, wakeup_cb, this);
  86. if (mpv_initialize(m_mpv) < 0)
  87. throw FatalException(tr("Failed to initialize mpv."));
  88. // Setup a hook with the ID 1, which is run during the file is loaded.
  89. // Used to delay playback start for display framerate switching.
  90. // (See handler in handleMpvEvent() for details.)
  91. mpv::qt::command_variant(m_mpv, QStringList() << "hook-add" << "on_load" << "1" << "0");
  92. updateAudioDeviceList();
  93. setAudioConfiguration();
  94. updateSubtitleSettings();
  95. updateVideoSettings();
  96. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_VIDEO), &SettingsSection::valuesUpdated,
  97. this, &PlayerComponent::updateVideoSettings);
  98. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_SUBTITLES), &SettingsSection::valuesUpdated,
  99. this, &PlayerComponent::updateSubtitleSettings);
  100. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_AUDIO), &SettingsSection::valuesUpdated,
  101. this, &PlayerComponent::setAudioConfiguration);
  102. return true;
  103. }
  104. ///////////////////////////////////////////////////////////////////////////////////////////////////
  105. void PlayerComponent::setQtQuickWindow(QQuickWindow* window)
  106. {
  107. PlayerQuickItem* video = window->findChild<PlayerQuickItem*>("video");
  108. if (!video)
  109. throw FatalException(tr("Failed to load video element."));
  110. mpv_set_option_string(m_mpv, "vo", "opengl-cb");
  111. video->initMpv(this);
  112. }
  113. ///////////////////////////////////////////////////////////////////////////////////////////////////
  114. void PlayerComponent::setRpiWindow(QQuickWindow* window)
  115. {
  116. window->setFlags(Qt::FramelessWindowHint);
  117. mpv_set_option_string(m_mpv, "vo", "rpi");
  118. }
  119. ///////////////////////////////////////////////////////////////////////////////////////////////////
  120. void PlayerComponent::setWindow(QQuickWindow* window)
  121. {
  122. bool use_rpi = false;
  123. #ifdef TARGET_RPI
  124. use_rpi = true;
  125. #endif
  126. m_window = window;
  127. if (!window)
  128. return;
  129. QString force_vo = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "debug.force_vo").toString();
  130. if (force_vo.size())
  131. mpv::qt::set_option_variant(m_mpv, "vo", force_vo);
  132. else if (use_rpi)
  133. setRpiWindow(window);
  134. else
  135. setQtQuickWindow(window);
  136. }
  137. ///////////////////////////////////////////////////////////////////////////////////////////////////
  138. bool PlayerComponent::load(const QString& url, const QVariantMap& options, const QVariantMap &metadata, const QString& audioStream , const QString& subtitleStream)
  139. {
  140. stop();
  141. queueMedia(url, options, metadata, audioStream, subtitleStream);
  142. return true;
  143. }
  144. ///////////////////////////////////////////////////////////////////////////////////////////////////
  145. void PlayerComponent::queueMedia(const QString& url, const QVariantMap& options, const QVariantMap &metadata, const QString& audioStream, const QString& subtitleStream)
  146. {
  147. m_mediaFrameRate = metadata["frameRate"].toFloat(); // returns 0 on failure
  148. updateVideoSettings();
  149. QVariantList command;
  150. command << "loadfile" << url;
  151. command << "append-play"; // if nothing is playing, play it now, otherwise just enqueue it
  152. QVariantMap extraArgs;
  153. quint64 startMilliseconds = options["startMilliseconds"].toLongLong();
  154. if (startMilliseconds != 0)
  155. extraArgs.insert("start", "+" + QString::number(startMilliseconds / 1000.0));
  156. // detect subtitles
  157. if (!subtitleStream.isEmpty())
  158. {
  159. // If the stream title starts with a #, then it's an index
  160. if (subtitleStream.startsWith("#"))
  161. extraArgs.insert("ff-sid", subtitleStream.mid(1));
  162. else
  163. extraArgs.insert("sub-file", subtitleStream);
  164. // try guessing the codepage. We could pass a country code to enca to help
  165. // it detect the correct one, but we don't get that from the server right now
  166. //
  167. extraArgs.insert("sub-codepage", "enca");
  168. }
  169. else
  170. {
  171. // no subtitles, tell mpv to ignore them.
  172. extraArgs.insert("sid", "no");
  173. }
  174. if (metadata["type"] == "music")
  175. extraArgs.insert("vid", "no");
  176. // and then the audio stream
  177. if (!audioStream.isEmpty())
  178. extraArgs.insert("ff-aid", audioStream);
  179. extraArgs.insert("pause", options["autoplay"].toBool() ? "no" : "yes");
  180. command << extraArgs;
  181. QLOG_DEBUG() << command;
  182. mpv::qt::command_variant(m_mpv, command);
  183. }
  184. ///////////////////////////////////////////////////////////////////////////////////////////////////
  185. bool PlayerComponent::switchDisplayFrameRate()
  186. {
  187. QLOG_DEBUG() << "Video framerate:" << m_mediaFrameRate << "fps";
  188. if (!SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "refreshrate.auto_switch").toBool())
  189. {
  190. QLOG_DEBUG() << "Not switching refresh-rate (disabled by settings).";
  191. return false;
  192. }
  193. bool fs = SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool();
  194. #if KONVERGO_OPENELEC
  195. fs = true;
  196. #endif
  197. if (!fs)
  198. {
  199. QLOG_DEBUG() << "Not switching refresh-rate (not in fullscreen mode).";
  200. return false;
  201. }
  202. if (m_mediaFrameRate < 1)
  203. {
  204. QLOG_DEBUG() << "Not switching refresh-rate (no known video framerate).";
  205. return false;
  206. }
  207. // Make sure a timer started by the previous file ending isn't accidentally
  208. // still in-flight. It could switch the display back after we've switched.
  209. m_restoreDisplayTimer.stop();
  210. DisplayComponent* display = &DisplayComponent::Get();
  211. if (!display->switchToBestVideoMode(m_mediaFrameRate))
  212. {
  213. QLOG_DEBUG() << "Switching refresh-rate failed or unnecessary.";
  214. return false;
  215. }
  216. // Make sure settings dependent on the display refresh rate are updated properly.
  217. updateVideoSettings();
  218. return true;
  219. }
  220. ///////////////////////////////////////////////////////////////////////////////////////////////////
  221. void PlayerComponent::onRestoreDisplay()
  222. {
  223. // If the player will in fact start another file (or is playing one), don't restore.
  224. if (mpv::qt::get_property_variant(m_mpv, "idle").toBool())
  225. DisplayComponent::Get().restorePreviousVideoMode();
  226. }
  227. ///////////////////////////////////////////////////////////////////////////////////////////////////
  228. void PlayerComponent::onRefreshRateChange()
  229. {
  230. // Make sure settings dependent on the display refresh rate are updated properly.
  231. updateVideoSettings();
  232. }
  233. ///////////////////////////////////////////////////////////////////////////////////////////////////
  234. void PlayerComponent::handleMpvEvent(mpv_event *event)
  235. {
  236. switch (event->event_id)
  237. {
  238. case MPV_EVENT_START_FILE:
  239. {
  240. m_CurrentUrl = mpv::qt::get_property_variant(m_mpv, "path").toString();
  241. m_playbackStartSent = false;
  242. break;
  243. }
  244. case MPV_EVENT_FILE_LOADED:
  245. {
  246. emit playing(m_CurrentUrl);
  247. break;
  248. }
  249. case MPV_EVENT_END_FILE:
  250. {
  251. mpv_event_end_file *end_file = (mpv_event_end_file *)event->data;
  252. switch (end_file->reason)
  253. {
  254. case MPV_END_FILE_REASON_EOF:
  255. emit finished(m_CurrentUrl);
  256. break;
  257. case MPV_END_FILE_REASON_ERROR:
  258. emit error(end_file->error, mpv_error_string(end_file->error));
  259. break;
  260. default:
  261. emit stopped(m_CurrentUrl);
  262. break;
  263. }
  264. emit playbackEnded(m_CurrentUrl);
  265. m_CurrentUrl = "";
  266. m_restoreDisplayTimer.start(0);
  267. break;
  268. }
  269. case MPV_EVENT_IDLE:
  270. {
  271. emit playbackAllDone();
  272. break;
  273. }
  274. case MPV_EVENT_PLAYBACK_RESTART:
  275. {
  276. // it's also sent after seeks are completed
  277. if (!m_playbackStartSent)
  278. emit playbackStarting();
  279. m_playbackStartSent = true;
  280. break;
  281. }
  282. case MPV_EVENT_PROPERTY_CHANGE:
  283. {
  284. mpv_event_property *prop = (mpv_event_property *)event->data;
  285. if (strcmp(prop->name, "pause") == 0 && prop->format == MPV_FORMAT_FLAG)
  286. {
  287. int state = *(int *)prop->data;
  288. emit paused(state);
  289. }
  290. else if (strcmp(prop->name, "cache-buffering-state") == 0 && prop->format == MPV_FORMAT_INT64)
  291. {
  292. int64_t percentage = *(int64_t *)prop->data;
  293. emit buffering(percentage);
  294. }
  295. else if (strcmp(prop->name, "playback-time") == 0 && prop->format == MPV_FORMAT_DOUBLE)
  296. {
  297. double pos = *(double*)prop->data;
  298. if (fabs(pos - m_lastPositionUpdate) > 0.25)
  299. {
  300. quint64 ms = (quint64)(qMax(pos * 1000.0, 0.0));
  301. emit positionUpdate(ms);
  302. m_lastPositionUpdate = pos;
  303. }
  304. }
  305. else if (strcmp(prop->name, "vo-configured") == 0)
  306. {
  307. int state = prop->format == MPV_FORMAT_FLAG ? *(int *)prop->data : 0;
  308. emit windowVisible(state);
  309. }
  310. else if (strcmp(prop->name, "duration") == 0)
  311. {
  312. if (prop->format == MPV_FORMAT_DOUBLE)
  313. emit updateDuration(*(double *)prop->data * 1000.0);
  314. }
  315. else if (strcmp(prop->name, "audio-device-list") == 0)
  316. {
  317. updateAudioDeviceList();
  318. }
  319. break;
  320. }
  321. case MPV_EVENT_LOG_MESSAGE:
  322. {
  323. mpv_event_log_message *msg = (mpv_event_log_message *)event->data;
  324. // Strip the trailing '\n'
  325. size_t len = strlen(msg->text);
  326. if (len > 0 && msg->text[len - 1] == '\n')
  327. len -= 1;
  328. QString logline = QString::fromUtf8(msg->prefix) + ": " + QString::fromUtf8(msg->text, len);
  329. if (msg->log_level >= MPV_LOG_LEVEL_V)
  330. QLOG_DEBUG() << qPrintable(logline);
  331. else if (msg->log_level >= MPV_LOG_LEVEL_INFO)
  332. QLOG_INFO() << qPrintable(logline);
  333. else if (msg->log_level >= MPV_LOG_LEVEL_WARN)
  334. QLOG_WARN() << qPrintable(logline);
  335. else
  336. QLOG_ERROR() << qPrintable(logline);
  337. break;
  338. }
  339. case MPV_EVENT_CLIENT_MESSAGE:
  340. {
  341. mpv_event_client_message *msg = (mpv_event_client_message *)event->data;
  342. // This happens when the player is about to load the file, but no actual loading has taken part yet.
  343. // We use this to block loading until we explicitly tell it to continue.
  344. if (msg->num_args >= 3 && !strcmp(msg->args[0], "hook_run") && !strcmp(msg->args[1], "1"))
  345. {
  346. QString resume_id = QString::fromUtf8(msg->args[2]);
  347. // Calling this lambda will instruct mpv to continue loading the file.
  348. auto resume = [=] {
  349. mpv::qt::command_variant(m_mpv, QStringList() << "hook-ack" << resume_id);
  350. };
  351. if (switchDisplayFrameRate())
  352. {
  353. // Now wait for some time for mode change - this is needed because mode changing can take some
  354. // time, during which the screen is black, and initializing hardware decoding could fail due
  355. // to various strange OS-related reasons.
  356. // (Better hope the user doesn't try to exit Konvergo during mode change.)
  357. int pause = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "refreshrate.delay").toInt() * 1000;
  358. QTimer::singleShot(pause, resume);
  359. }
  360. else
  361. {
  362. resume();
  363. }
  364. break;
  365. }
  366. }
  367. default:; /* ignore */
  368. }
  369. }
  370. ///////////////////////////////////////////////////////////////////////////////////////////////////
  371. void PlayerComponent::handleMpvEvents()
  372. {
  373. // Process all events, until the event queue is empty.
  374. while (1)
  375. {
  376. mpv_event *event = mpv_wait_event(m_mpv, 0);
  377. if (event->event_id == MPV_EVENT_NONE)
  378. break;
  379. handleMpvEvent(event);
  380. }
  381. }
  382. ///////////////////////////////////////////////////////////////////////////////////////////////////
  383. void PlayerComponent::setVideoOnlyMode(bool enable)
  384. {
  385. if (m_window)
  386. {
  387. QQuickItem *web = m_window->findChild<QQuickItem *>("web");
  388. if (web)
  389. web->setVisible(!enable);
  390. }
  391. }
  392. ///////////////////////////////////////////////////////////////////////////////////////////////////
  393. void PlayerComponent::play()
  394. {
  395. QStringList args = (QStringList() << "set" << "pause" << "no");
  396. mpv::qt::command_variant(m_mpv, args);
  397. }
  398. ///////////////////////////////////////////////////////////////////////////////////////////////////
  399. void PlayerComponent::stop()
  400. {
  401. QStringList args("stop");
  402. mpv::qt::command_variant(m_mpv, args);
  403. }
  404. ///////////////////////////////////////////////////////////////////////////////////////////////////
  405. void PlayerComponent::clearQueue()
  406. {
  407. QStringList args("playlist_clear");
  408. mpv::qt::command_variant(m_mpv, args);
  409. }
  410. ///////////////////////////////////////////////////////////////////////////////////////////////////
  411. void PlayerComponent::pause()
  412. {
  413. QStringList args = (QStringList() << "set" << "pause" << "yes");
  414. mpv::qt::command_variant(m_mpv, args);
  415. }
  416. ///////////////////////////////////////////////////////////////////////////////////////////////////
  417. void PlayerComponent::seekTo(qint64 ms)
  418. {
  419. double start = mpv::qt::get_property_variant(m_mpv, "time-start").toDouble();
  420. QString timeStr = QString::number(ms / 1000.0 + start);
  421. QStringList args = (QStringList() << "seek" << timeStr << "absolute");
  422. mpv::qt::command_variant(m_mpv, args);
  423. }
  424. ///////////////////////////////////////////////////////////////////////////////////////////////////
  425. QVariant PlayerComponent::getAudioDeviceList()
  426. {
  427. return mpv::qt::get_property_variant(m_mpv, "audio-device-list");
  428. }
  429. ///////////////////////////////////////////////////////////////////////////////////////////////////
  430. void PlayerComponent::setAudioDevice(const QString& name)
  431. {
  432. mpv::qt::set_property_variant(m_mpv, "audio-device", name);
  433. }
  434. ///////////////////////////////////////////////////////////////////////////////////////////////////
  435. void PlayerComponent::setVolume(quint8 volume)
  436. {
  437. // Will fail if no audio output opened (i.e. no file playing)
  438. mpv::qt::set_property_variant(m_mpv, "volume", volume);
  439. }
  440. ///////////////////////////////////////////////////////////////////////////////////////////////////
  441. quint8 PlayerComponent::volume()
  442. {
  443. QVariant volume = mpv::qt::get_property_variant(m_mpv, "volume");
  444. if (volume.isValid())
  445. return volume.toInt();
  446. return 0;
  447. }
  448. ///////////////////////////////////////////////////////////////////////////////////////////////////
  449. void PlayerComponent::setSubtitleStream(const QString &subtitleStream)
  450. {
  451. if (subtitleStream.isEmpty())
  452. {
  453. mpv::qt::set_property_variant(m_mpv, "ff-sid", "no");
  454. }
  455. else if (subtitleStream.startsWith("#"))
  456. {
  457. mpv::qt::set_property_variant(m_mpv, "ff-sid", subtitleStream.mid(1));
  458. }
  459. else
  460. {
  461. QStringList args = (QStringList() << "sub_add" << subtitleStream << "cached");
  462. mpv::qt::command_variant(m_mpv, args);
  463. }
  464. }
  465. ///////////////////////////////////////////////////////////////////////////////////////////////////
  466. void PlayerComponent::setAudioStream(const QString &audioStream)
  467. {
  468. if (audioStream.isEmpty())
  469. mpv::qt::set_property_variant(m_mpv, "ff-aid", "no");
  470. else
  471. mpv::qt::set_property_variant(m_mpv, "ff-aid", audioStream);
  472. }
  473. /////////////////////////////////////////////////////////////////////////////////////////
  474. void PlayerComponent::setAudioDelay(qint64 milliseconds)
  475. {
  476. m_playbackAudioDelay = milliseconds;
  477. double display_fps = DisplayComponent::Get().currentRefreshRate();
  478. const char *audio_delay_setting = "audio_delay.normal";
  479. if (fabs(display_fps - 24) < 1) // cover 24Hz, 23.976Hz, and values very close
  480. audio_delay_setting = "audio_delay.24hz";
  481. double fixed_delay = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, audio_delay_setting).toFloat();
  482. mpv::qt::set_option_variant(m_mpv, "audio-delay", (fixed_delay + m_playbackAudioDelay) / 1000.0);
  483. }
  484. /////////////////////////////////////////////////////////////////////////////////////////
  485. void PlayerComponent::setSubtitleDelay(qint64 milliseconds)
  486. {
  487. mpv::qt::set_option_variant(m_mpv, "sub-delay", milliseconds / 1000.0);
  488. }
  489. ///////////////////////////////////////////////////////////////////////////////////////////////////
  490. void PlayerComponent::onReloadAudio()
  491. {
  492. mpv::qt::command_variant(m_mpv, QStringList() << "ao_reload");
  493. }
  494. ///////////////////////////////////////////////////////////////////////////////////////////////////
  495. // This is called with the set of previous audio devices that were detected, and the set of current
  496. // audio devices. From this we guess whether we should reopen the audio device. If the user-selected
  497. // device went away previously, and now comes back, reinitializing the player's audio output will
  498. // force the player and/or the OS to move audio output back to the user-selected device.
  499. void PlayerComponent::checkCurrentAudioDevice(const QSet<QString>& old_devs, const QSet<QString>& new_devs)
  500. {
  501. QString user_device = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device").toString();
  502. QSet<QString> removed = old_devs - new_devs;
  503. QSet<QString> added = new_devs - old_devs;
  504. QLOG_DEBUG() << "Audio devices removed:" << removed;
  505. QLOG_DEBUG() << "Audio devices added:" << added;
  506. QLOG_DEBUG() << "Audio device selected:" << user_device;
  507. if (!mpv::qt::get_property_variant(m_mpv, "idle").toBool() && user_device.length())
  508. {
  509. if (added.contains(user_device))
  510. {
  511. // The timer is for debouncing the reload. Several change notifications could
  512. // come in quick succession. Also, it's possible that trying to open the
  513. // reappeared audio device immediately can fail.
  514. m_reloadAudioTimer.start(500);
  515. }
  516. }
  517. }
  518. ///////////////////////////////////////////////////////////////////////////////////////////////////
  519. void PlayerComponent::updateAudioDeviceList()
  520. {
  521. QVariantList settingList;
  522. QVariant list = getAudioDeviceList();
  523. QSet<QString> devices;
  524. foreach (const QVariant& d, list.toList())
  525. {
  526. Q_ASSERT(d.type() == QVariant::Map);
  527. QVariantMap dmap = d.toMap();
  528. devices.insert(dmap["name"].toString());
  529. QVariantMap entry;
  530. entry["value"] = dmap["name"];
  531. entry["title"] = dmap["description"];
  532. settingList << entry;
  533. }
  534. SettingsComponent::Get().updatePossibleValues(SETTINGS_SECTION_AUDIO, "device", settingList);
  535. checkCurrentAudioDevice(m_audioDevices, devices);
  536. m_audioDevices = devices;
  537. }
  538. ///////////////////////////////////////////////////////////////////////////////////////////////////
  539. void PlayerComponent::setAudioConfiguration()
  540. {
  541. QStringList ao_defaults;
  542. // On OSX, ask mpv to change the audio format system-wide. This is needed
  543. // at least to get multichannel PCM running, if the user didn't already
  544. // select multichannel output in "Audio MIDI Setup".
  545. ao_defaults << "coreaudio:change-physical-format=yes";
  546. QString deviceType = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "devicetype").toString();
  547. if (SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "exclusive").toBool())
  548. {
  549. ao_defaults << "wasapi:exclusive=yes";
  550. ao_defaults << "coreaudio:exclusive=yes";
  551. }
  552. mpv::qt::set_option_variant(m_mpv, "ao-defaults", ao_defaults.join(','));
  553. // set the audio device
  554. QVariant device = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device");
  555. mpv::qt::set_property_variant(m_mpv, "audio-device", device);
  556. // set the channel layout
  557. QVariant layout = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "channels");
  558. mpv::qt::set_option_variant(m_mpv, "audio-channels", layout);
  559. QString resampleOpts = "";
  560. bool normalize = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "normalize").toBool();
  561. resampleOpts += QString(":normalize=") + (normalize ? "yes" : "no");
  562. mpv::qt::set_option_variant(m_mpv, "af-defaults", "lavrresample" + resampleOpts);
  563. QString passthroughCodecs;
  564. // passthrough doesn't make sense with basic type
  565. if (deviceType != AUDIO_DEVICE_TYPE_BASIC)
  566. {
  567. QStringList enabledCodecs;
  568. SettingsSection* audioSection = SettingsComponent::Get().getSection(SETTINGS_SECTION_AUDIO);
  569. QStringList codecs;
  570. if (deviceType == AUDIO_DEVICE_TYPE_SPDIF)
  571. codecs = AudioCodecsSPDIF();
  572. else if (deviceType == AUDIO_DEVICE_TYPE_HDMI && audioSection->value("advanced").toBool())
  573. codecs = AudioCodecsAll();
  574. foreach (const QString& key, codecs)
  575. {
  576. if (audioSection->value("passthrough." + key).toBool())
  577. enabledCodecs << key;
  578. }
  579. // dts-hd includes dts, but listing dts before dts-hd may disable dts-hd.
  580. if (enabledCodecs.indexOf("dts-hd") != -1)
  581. enabledCodecs.removeAll("dts");
  582. passthroughCodecs = enabledCodecs.join(",");
  583. }
  584. mpv::qt::set_option_variant(m_mpv, "audio-spdif", passthroughCodecs);
  585. // if the user has indicated that we have a optical spdif connection
  586. // we need to set this extra option that allows us to transcode
  587. // 5.1 audio into a usable format, note that we only support AC3
  588. // here for now. We might need to add support for DTS transcoding
  589. // if we see user requests for it.
  590. //
  591. bool doAc3Transcoding = false;
  592. if (deviceType == AUDIO_DEVICE_TYPE_SPDIF &&
  593. SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "passthrough.ac3").toBool())
  594. {
  595. QLOG_INFO() << "Enabling audio AC3 transcoding (if needed)";
  596. mpv::qt::set_option_variant(m_mpv, "af", "lavcac3enc");
  597. doAc3Transcoding = true;
  598. }
  599. // Make a informational log message.
  600. QString audioConfig = QString(QString("Audio Config - device: %1, ") +
  601. "channel layout: %2, " +
  602. "passthrough codecs: %3, " +
  603. "ac3 transcoding: %4").arg(device.toString(),
  604. layout.toString(),
  605. passthroughCodecs.isEmpty() ? "none" : passthroughCodecs,
  606. doAc3Transcoding ? "yes" : "no");
  607. QLOG_INFO() << qPrintable(audioConfig);
  608. }
  609. ///////////////////////////////////////////////////////////////////////////////////////////////////
  610. void PlayerComponent::updateSubtitleSettings()
  611. {
  612. QVariant size = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "size");
  613. mpv::qt::set_option_variant(m_mpv, "sub-text-font-size", size);
  614. QVariant colors_string = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "color");
  615. auto colors = colors_string.toString().split(",");
  616. if (colors.length() == 2)
  617. {
  618. mpv::qt::set_option_variant(m_mpv, "sub-text-color", colors[0]);
  619. mpv::qt::set_option_variant(m_mpv, "sub-text-border-color", colors[1]);
  620. }
  621. QVariant subpos_string = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "placement");
  622. auto subpos = subpos_string.toString().split(",");
  623. if (subpos.length() == 2)
  624. {
  625. mpv::qt::set_option_variant(m_mpv, "sub-text-align-x", subpos[0]);
  626. mpv::qt::set_option_variant(m_mpv, "sub-text-align-y", subpos[1]);
  627. }
  628. }
  629. ///////////////////////////////////////////////////////////////////////////////////////////////////
  630. void PlayerComponent::updateVideoSettings()
  631. {
  632. QVariant sync_mode = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "sync_mode");
  633. mpv::qt::set_option_variant(m_mpv, "video-sync", sync_mode);
  634. QVariant hardware_decoding = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "hardware_decoding");
  635. mpv::qt::set_property_variant(m_mpv, "hwdec", hardware_decoding.toBool() ? "auto" : "no");
  636. QVariant deinterlace = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "deinterlace");
  637. mpv::qt::set_option_variant(m_mpv, "deinterlace", deinterlace.toBool() ? "yes" : "no");
  638. #ifndef TARGET_RPI
  639. double display_fps = DisplayComponent::Get().currentRefreshRate();
  640. mpv::qt::set_option_variant(m_mpv, "display-fps", display_fps);
  641. #endif
  642. setAudioDelay(m_playbackAudioDelay);
  643. QVariant cache = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "cache");
  644. mpv::qt::set_option_variant(m_mpv, "cache", cache.toInt() * 1024);
  645. }
  646. /////////////////////////////////////////////////////////////////////////////////////////
  647. void PlayerComponent::userCommand(const QString& command)
  648. {
  649. QByteArray cmd_utf8 = command.toUtf8();
  650. mpv_command_string(m_mpv, cmd_utf8.data());
  651. }
  652. /////////////////////////////////////////////////////////////////////////////////////////
  653. static QString get_mpv_osd(mpv_handle *ctx, const char *property)
  654. {
  655. char *s = mpv_get_property_osd_string(ctx, property);
  656. if (!s)
  657. return "-";
  658. QString r = QString::fromUtf8(s);
  659. mpv_free(s);
  660. return r;
  661. }
  662. #define MPV_PROPERTY(p) get_mpv_osd(m_mpv, p)
  663. /////////////////////////////////////////////////////////////////////////////////////////
  664. QString PlayerComponent::videoInformation() const
  665. {
  666. QString infoStr;
  667. QTextStream info(&infoStr);
  668. // check if video is playing
  669. if (mpv::qt::get_property_variant(m_mpv, "idle").toBool())
  670. return "";
  671. info << "File:" << endl;
  672. info << "URL: " << MPV_PROPERTY("path") << endl;
  673. info << "Container: " << MPV_PROPERTY("file-format") << endl;
  674. info << endl;
  675. info << "Video:" << endl;
  676. info << "Codec: " << MPV_PROPERTY("video-codec") << endl;
  677. info << "Size: " << MPV_PROPERTY("video-params/dw") << "x"
  678. << MPV_PROPERTY("video-params/dh") << endl;
  679. info << "FPS: " << MPV_PROPERTY("fps") << endl;
  680. info << "Aspect: " << MPV_PROPERTY("video-aspect") << endl;
  681. info << "Bitrate: " << MPV_PROPERTY("video-bitrate") << endl;
  682. info << "Display Sync: " << MPV_PROPERTY("display-sync-active") << endl;
  683. info << "Hardware Decoding: " << MPV_PROPERTY("hwdec-active")
  684. << " (" << MPV_PROPERTY("hwdec-detected") << ")" << endl;
  685. info << endl;
  686. info << "Audio: " << endl;
  687. info << "Codec: " << MPV_PROPERTY("audio-codec") << endl;
  688. info << "Bitrate: " << MPV_PROPERTY("audio-bitrate") << endl;
  689. info << "Channels (input): " << MPV_PROPERTY("audio-params/channels") << endl;
  690. info << "Channels (output): " << MPV_PROPERTY("audio-out-params/hr-channels")
  691. << " (" << MPV_PROPERTY("audio-out-params/channels") << ")" << endl;
  692. info << endl;
  693. info << "Performance: " << endl;
  694. info << "A/V: " << MPV_PROPERTY("avsync") << endl;
  695. info << "Change: " << MPV_PROPERTY("total-avsync-change") << endl;
  696. info << "Dropped frames: " << MPV_PROPERTY("vo-drop-frame-count") << endl;
  697. info << "Missed frames: " << MPV_PROPERTY("vo-missed-frame-count") << endl;
  698. info << endl;
  699. info << "Cache:" << endl;
  700. info << "Seconds: " << MPV_PROPERTY("demuxer-cache-duration") << endl;
  701. info << "Extra readahead: " << MPV_PROPERTY("cache-used") << endl;
  702. info << "Buffering: " << MPV_PROPERTY("cache-buffering-state") << endl;
  703. info << endl;
  704. info << "Misc: " << endl;
  705. info << "Time: " << MPV_PROPERTY("playback-time") << " / "
  706. << MPV_PROPERTY("duration")
  707. << " (" << MPV_PROPERTY("percent-pos") << "%)" << endl;
  708. info << "Pause state: " << MPV_PROPERTY("pause") << " / "
  709. << MPV_PROPERTY("paused-for-cache") << " / "
  710. << MPV_PROPERTY("core-idle") << endl;
  711. info << "Seeking: " << MPV_PROPERTY("seeking") << endl;
  712. info << "Seekable: " << MPV_PROPERTY("seekable") << " / "
  713. << MPV_PROPERTY("partially-seekable") << endl;
  714. info << flush;
  715. return infoStr;
  716. }