#include "PlayerComponent.h" #include #include #include #include #include "display/DisplayComponent.h" #include "settings/SettingsComponent.h" #include "system/SystemComponent.h" #include "utils/Utils.h" #include "ComponentManager.h" #include "settings/SettingsSection.h" #include "PlayerQuickItem.h" #include "QsLog.h" #include #include #include /////////////////////////////////////////////////////////////////////////////////////////////////// static void wakeup_cb(void *context) { PlayerComponent *player = (PlayerComponent *)context; emit player->onMpvEvents(); } /////////////////////////////////////////////////////////////////////////////////////////////////// PlayerComponent::PlayerComponent(QObject* parent) : ComponentBase(parent), m_lastPositionUpdate(0.0), m_playbackAudioDelay(0), m_playbackStartSent(false), m_window(0), m_mediaFrameRate(0), m_restoreDisplayTimer(this), m_reloadAudioTimer(this) { qmlRegisterType("Konvergo", 1, 0, "MpvVideo"); // deprecated name qmlRegisterType("Konvergo", 1, 0, "KonvergoVideo"); m_restoreDisplayTimer.setSingleShot(true); connect(&m_restoreDisplayTimer, &QTimer::timeout, this, &PlayerComponent::onRestoreDisplay); connect(&DisplayComponent::Get(), &DisplayComponent::refreshRateChanged, this, &PlayerComponent::onRefreshRateChange); m_reloadAudioTimer.setSingleShot(true); connect(&m_reloadAudioTimer, &QTimer::timeout, this, &PlayerComponent::onReloadAudio); } /////////////////////////////////////////////////////////////////////////////////////////////////// PlayerComponent::~PlayerComponent() { if (m_mpv) mpv_set_wakeup_callback(m_mpv, NULL, NULL); } /////////////////////////////////////////////////////////////////////////////////////////////////// bool PlayerComponent::componentInitialize() { m_mpv = mpv::qt::Handle::FromRawHandle(mpv_create()); if (!m_mpv) throw FatalException(tr("Failed to load mpv.")); mpv_request_log_messages(m_mpv, "terminal-default"); mpv_set_option_string(m_mpv, "msg-level", "all=v"); // No mouse events mpv_set_option_string(m_mpv, "input-cursor", "no"); mpv_set_option_string(m_mpv, "cursor-autohide", "no"); mpv_set_option_string(m_mpv, "config", "yes"); mpv_set_option_string(m_mpv, "config-dir", Paths::dataDir().toUtf8().data()); // We don't need this, so avoid initializing fontconfig. mpv_set_option_string(m_mpv, "use-text-osd", "no"); // Always use the system mixer. mpv_set_option_string(m_mpv, "softvol", "no"); // Just discard audio output if no audio device could be opened. This gives // us better flexibility how to react to such errors (instead of just // aborting playback immediately). mpv_set_option_string(m_mpv, "audio-fallback-to-null", "yes"); // Make it load the hwdec interop, so hwdec can be enabled at runtime. mpv::qt::set_option_variant(m_mpv, "hwdec-preload", "auto"); // User-visible application name used by some audio APIs (at least PulseAudio). mpv_set_option_string(m_mpv, "audio-client-name", QCoreApplication::applicationName().toUtf8().data()); // User-visible stream title used by some audio APIs (at least PulseAudio and wasapi). mpv_set_option_string(m_mpv, "title", QCoreApplication::applicationName().toUtf8().data()); // Apply some low-memory settings on RPI, which is relatively memory-constrained. #ifdef TARGET_RPI // The backbuffer makes seeking back faster (without having to do a HTTP-level seek) mpv::qt::set_option_variant(m_mpv, "cache-backbuffer", 10 * 1024); // KB // The demuxer queue is used for the readahead, and also for dealing with badly // interlaved audio/video. Setting it too low increases sensitivity to network // issues, and could cause playback failure with "bad" files. mpv::qt::set_option_variant(m_mpv, "demuxer-max-bytes", 50 * 1024 * 1024); // bytes #endif mpv_observe_property(m_mpv, 0, "pause", MPV_FORMAT_FLAG); mpv_observe_property(m_mpv, 0, "cache-buffering-state", MPV_FORMAT_INT64); mpv_observe_property(m_mpv, 0, "playback-time", MPV_FORMAT_DOUBLE); mpv_observe_property(m_mpv, 0, "vo-configured", MPV_FORMAT_FLAG); mpv_observe_property(m_mpv, 0, "duration", MPV_FORMAT_DOUBLE); mpv_observe_property(m_mpv, 0, "audio-device-list", MPV_FORMAT_NODE); connect(this, &PlayerComponent::onMpvEvents, this, &PlayerComponent::handleMpvEvents, Qt::QueuedConnection); mpv_set_wakeup_callback(m_mpv, wakeup_cb, this); if (mpv_initialize(m_mpv) < 0) throw FatalException(tr("Failed to initialize mpv.")); // Setup a hook with the ID 1, which is run during the file is loaded. // Used to delay playback start for display framerate switching. // (See handler in handleMpvEvent() for details.) mpv::qt::command_variant(m_mpv, QStringList() << "hook-add" << "on_load" << "1" << "0"); updateAudioDeviceList(); setAudioConfiguration(); updateSubtitleSettings(); updateVideoSettings(); connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_VIDEO), &SettingsSection::valuesUpdated, this, &PlayerComponent::updateVideoSettings); connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_SUBTITLES), &SettingsSection::valuesUpdated, this, &PlayerComponent::updateSubtitleSettings); connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_AUDIO), &SettingsSection::valuesUpdated, this, &PlayerComponent::setAudioConfiguration); return true; } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setQtQuickWindow(QQuickWindow* window) { PlayerQuickItem* video = window->findChild("video"); if (!video) throw FatalException(tr("Failed to load video element.")); mpv_set_option_string(m_mpv, "vo", "opengl-cb"); video->initMpv(this); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setRpiWindow(QQuickWindow* window) { window->setFlags(Qt::FramelessWindowHint); mpv_set_option_string(m_mpv, "vo", "rpi"); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setWindow(QQuickWindow* window) { bool use_rpi = false; #ifdef TARGET_RPI use_rpi = true; #endif m_window = window; if (!window) return; QString force_vo = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "debug.force_vo").toString(); if (force_vo.size()) mpv::qt::set_option_variant(m_mpv, "vo", force_vo); else if (use_rpi) setRpiWindow(window); else setQtQuickWindow(window); } /////////////////////////////////////////////////////////////////////////////////////////////////// bool PlayerComponent::load(const QString& url, const QVariantMap& options, const QVariantMap &metadata, const QString& audioStream , const QString& subtitleStream) { stop(); queueMedia(url, options, metadata, audioStream, subtitleStream); return true; } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::queueMedia(const QString& url, const QVariantMap& options, const QVariantMap &metadata, const QString& audioStream, const QString& subtitleStream) { m_mediaFrameRate = metadata["frameRate"].toFloat(); // returns 0 on failure updateVideoSettings(); QVariantList command; command << "loadfile" << url; command << "append-play"; // if nothing is playing, play it now, otherwise just enqueue it QVariantMap extraArgs; quint64 startMilliseconds = options["startMilliseconds"].toLongLong(); if (startMilliseconds != 0) extraArgs.insert("start", "+" + QString::number(startMilliseconds / 1000.0)); // detect subtitles if (!subtitleStream.isEmpty()) { // If the stream title starts with a #, then it's an index if (subtitleStream.startsWith("#")) extraArgs.insert("ff-sid", subtitleStream.mid(1)); else extraArgs.insert("sub-file", subtitleStream); // try guessing the codepage. We could pass a country code to enca to help // it detect the correct one, but we don't get that from the server right now // extraArgs.insert("sub-codepage", "enca"); } else { // no subtitles, tell mpv to ignore them. extraArgs.insert("sid", "no"); } if (metadata["type"] == "music") extraArgs.insert("vid", "no"); // and then the audio stream if (!audioStream.isEmpty()) extraArgs.insert("ff-aid", audioStream); extraArgs.insert("pause", options["autoplay"].toBool() ? "no" : "yes"); command << extraArgs; QLOG_DEBUG() << command; mpv::qt::command_variant(m_mpv, command); } /////////////////////////////////////////////////////////////////////////////////////////////////// bool PlayerComponent::switchDisplayFrameRate() { QLOG_DEBUG() << "Video framerate:" << m_mediaFrameRate << "fps"; if (!SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "refreshrate.auto_switch").toBool()) { QLOG_DEBUG() << "Not switching refresh-rate (disabled by settings)."; return false; } bool fs = SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool(); #if KONVERGO_OPENELEC fs = true; #endif if (!fs) { QLOG_DEBUG() << "Not switching refresh-rate (not in fullscreen mode)."; return false; } if (m_mediaFrameRate < 1) { QLOG_DEBUG() << "Not switching refresh-rate (no known video framerate)."; return false; } // Make sure a timer started by the previous file ending isn't accidentally // still in-flight. It could switch the display back after we've switched. m_restoreDisplayTimer.stop(); DisplayComponent* display = &DisplayComponent::Get(); if (!display->switchToBestVideoMode(m_mediaFrameRate)) { QLOG_DEBUG() << "Switching refresh-rate failed or unnecessary."; return false; } // Make sure settings dependent on the display refresh rate are updated properly. updateVideoSettings(); return true; } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::onRestoreDisplay() { // If the player will in fact start another file (or is playing one), don't restore. if (mpv::qt::get_property_variant(m_mpv, "idle").toBool()) DisplayComponent::Get().restorePreviousVideoMode(); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::onRefreshRateChange() { // Make sure settings dependent on the display refresh rate are updated properly. updateVideoSettings(); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::handleMpvEvent(mpv_event *event) { switch (event->event_id) { case MPV_EVENT_START_FILE: { m_CurrentUrl = mpv::qt::get_property_variant(m_mpv, "path").toString(); m_playbackStartSent = false; break; } case MPV_EVENT_FILE_LOADED: { emit playing(m_CurrentUrl); break; } case MPV_EVENT_END_FILE: { mpv_event_end_file *end_file = (mpv_event_end_file *)event->data; switch (end_file->reason) { case MPV_END_FILE_REASON_EOF: emit finished(m_CurrentUrl); break; case MPV_END_FILE_REASON_ERROR: emit error(end_file->error, mpv_error_string(end_file->error)); break; default: emit stopped(m_CurrentUrl); break; } emit playbackEnded(m_CurrentUrl); m_CurrentUrl = ""; m_restoreDisplayTimer.start(0); break; } case MPV_EVENT_IDLE: { emit playbackAllDone(); break; } case MPV_EVENT_PLAYBACK_RESTART: { // it's also sent after seeks are completed if (!m_playbackStartSent) emit playbackStarting(); m_playbackStartSent = true; break; } case MPV_EVENT_PROPERTY_CHANGE: { mpv_event_property *prop = (mpv_event_property *)event->data; if (strcmp(prop->name, "pause") == 0 && prop->format == MPV_FORMAT_FLAG) { int state = *(int *)prop->data; emit paused(state); } else if (strcmp(prop->name, "cache-buffering-state") == 0 && prop->format == MPV_FORMAT_INT64) { int64_t percentage = *(int64_t *)prop->data; emit buffering(percentage); } else if (strcmp(prop->name, "playback-time") == 0 && prop->format == MPV_FORMAT_DOUBLE) { double pos = *(double*)prop->data; if (fabs(pos - m_lastPositionUpdate) > 0.25) { quint64 ms = (quint64)(qMax(pos * 1000.0, 0.0)); emit positionUpdate(ms); m_lastPositionUpdate = pos; } } else if (strcmp(prop->name, "vo-configured") == 0) { int state = prop->format == MPV_FORMAT_FLAG ? *(int *)prop->data : 0; emit windowVisible(state); } else if (strcmp(prop->name, "duration") == 0) { if (prop->format == MPV_FORMAT_DOUBLE) emit updateDuration(*(double *)prop->data * 1000.0); } else if (strcmp(prop->name, "audio-device-list") == 0) { updateAudioDeviceList(); } break; } case MPV_EVENT_LOG_MESSAGE: { mpv_event_log_message *msg = (mpv_event_log_message *)event->data; // Strip the trailing '\n' size_t len = strlen(msg->text); if (len > 0 && msg->text[len - 1] == '\n') len -= 1; QString logline = QString::fromUtf8(msg->prefix) + ": " + QString::fromUtf8(msg->text, len); if (msg->log_level >= MPV_LOG_LEVEL_V) QLOG_DEBUG() << qPrintable(logline); else if (msg->log_level >= MPV_LOG_LEVEL_INFO) QLOG_INFO() << qPrintable(logline); else if (msg->log_level >= MPV_LOG_LEVEL_WARN) QLOG_WARN() << qPrintable(logline); else QLOG_ERROR() << qPrintable(logline); break; } case MPV_EVENT_CLIENT_MESSAGE: { mpv_event_client_message *msg = (mpv_event_client_message *)event->data; // This happens when the player is about to load the file, but no actual loading has taken part yet. // We use this to block loading until we explicitly tell it to continue. if (msg->num_args >= 3 && !strcmp(msg->args[0], "hook_run") && !strcmp(msg->args[1], "1")) { QString resume_id = QString::fromUtf8(msg->args[2]); // Calling this lambda will instruct mpv to continue loading the file. auto resume = [=] { mpv::qt::command_variant(m_mpv, QStringList() << "hook-ack" << resume_id); }; if (switchDisplayFrameRate()) { // Now wait for some time for mode change - this is needed because mode changing can take some // time, during which the screen is black, and initializing hardware decoding could fail due // to various strange OS-related reasons. // (Better hope the user doesn't try to exit Konvergo during mode change.) int pause = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "refreshrate.delay").toInt() * 1000; QTimer::singleShot(pause, resume); } else { resume(); } break; } } default:; /* ignore */ } } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::handleMpvEvents() { // Process all events, until the event queue is empty. while (1) { mpv_event *event = mpv_wait_event(m_mpv, 0); if (event->event_id == MPV_EVENT_NONE) break; handleMpvEvent(event); } } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setVideoOnlyMode(bool enable) { if (m_window) { QQuickItem *web = m_window->findChild("web"); if (web) web->setVisible(!enable); } } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::play() { QStringList args = (QStringList() << "set" << "pause" << "no"); mpv::qt::command_variant(m_mpv, args); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::stop() { QStringList args("stop"); mpv::qt::command_variant(m_mpv, args); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::clearQueue() { QStringList args("playlist_clear"); mpv::qt::command_variant(m_mpv, args); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::pause() { QStringList args = (QStringList() << "set" << "pause" << "yes"); mpv::qt::command_variant(m_mpv, args); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::seekTo(qint64 ms) { double start = mpv::qt::get_property_variant(m_mpv, "time-start").toDouble(); QString timeStr = QString::number(ms / 1000.0 + start); QStringList args = (QStringList() << "seek" << timeStr << "absolute"); mpv::qt::command_variant(m_mpv, args); } /////////////////////////////////////////////////////////////////////////////////////////////////// QVariant PlayerComponent::getAudioDeviceList() { return mpv::qt::get_property_variant(m_mpv, "audio-device-list"); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setAudioDevice(const QString& name) { mpv::qt::set_property_variant(m_mpv, "audio-device", name); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setVolume(quint8 volume) { // Will fail if no audio output opened (i.e. no file playing) mpv::qt::set_property_variant(m_mpv, "volume", volume); } /////////////////////////////////////////////////////////////////////////////////////////////////// quint8 PlayerComponent::volume() { QVariant volume = mpv::qt::get_property_variant(m_mpv, "volume"); if (volume.isValid()) return volume.toInt(); return 0; } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setSubtitleStream(const QString &subtitleStream) { if (subtitleStream.isEmpty()) { mpv::qt::set_property_variant(m_mpv, "ff-sid", "no"); } else if (subtitleStream.startsWith("#")) { mpv::qt::set_property_variant(m_mpv, "ff-sid", subtitleStream.mid(1)); } else { QStringList args = (QStringList() << "sub_add" << subtitleStream << "cached"); mpv::qt::command_variant(m_mpv, args); } } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setAudioStream(const QString &audioStream) { if (audioStream.isEmpty()) mpv::qt::set_property_variant(m_mpv, "ff-aid", "no"); else mpv::qt::set_property_variant(m_mpv, "ff-aid", audioStream); } ///////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setAudioDelay(qint64 milliseconds) { m_playbackAudioDelay = milliseconds; double display_fps = DisplayComponent::Get().currentRefreshRate(); const char *audio_delay_setting = "audio_delay.normal"; if (fabs(display_fps - 24) < 1) // cover 24Hz, 23.976Hz, and values very close audio_delay_setting = "audio_delay.24hz"; double fixed_delay = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, audio_delay_setting).toFloat(); mpv::qt::set_option_variant(m_mpv, "audio-delay", (fixed_delay + m_playbackAudioDelay) / 1000.0); } ///////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setSubtitleDelay(qint64 milliseconds) { mpv::qt::set_option_variant(m_mpv, "sub-delay", milliseconds / 1000.0); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::onReloadAudio() { mpv::qt::command_variant(m_mpv, QStringList() << "ao_reload"); } /////////////////////////////////////////////////////////////////////////////////////////////////// // This is called with the set of previous audio devices that were detected, and the set of current // audio devices. From this we guess whether we should reopen the audio device. If the user-selected // device went away previously, and now comes back, reinitializing the player's audio output will // force the player and/or the OS to move audio output back to the user-selected device. void PlayerComponent::checkCurrentAudioDevice(const QSet& old_devs, const QSet& new_devs) { QString user_device = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device").toString(); QSet removed = old_devs - new_devs; QSet added = new_devs - old_devs; QLOG_DEBUG() << "Audio devices removed:" << removed; QLOG_DEBUG() << "Audio devices added:" << added; QLOG_DEBUG() << "Audio device selected:" << user_device; if (!mpv::qt::get_property_variant(m_mpv, "idle").toBool() && user_device.length()) { if (added.contains(user_device)) { // The timer is for debouncing the reload. Several change notifications could // come in quick succession. Also, it's possible that trying to open the // reappeared audio device immediately can fail. m_reloadAudioTimer.start(500); } } } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::updateAudioDeviceList() { QVariantList settingList; QVariant list = getAudioDeviceList(); QSet devices; foreach (const QVariant& d, list.toList()) { Q_ASSERT(d.type() == QVariant::Map); QVariantMap dmap = d.toMap(); devices.insert(dmap["name"].toString()); QVariantMap entry; entry["value"] = dmap["name"]; entry["title"] = dmap["description"]; settingList << entry; } SettingsComponent::Get().updatePossibleValues(SETTINGS_SECTION_AUDIO, "device", settingList); checkCurrentAudioDevice(m_audioDevices, devices); m_audioDevices = devices; } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::setAudioConfiguration() { QStringList ao_defaults; // On OSX, ask mpv to change the audio format system-wide. This is needed // at least to get multichannel PCM running, if the user didn't already // select multichannel output in "Audio MIDI Setup". ao_defaults << "coreaudio:change-physical-format=yes"; QString deviceType = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "devicetype").toString(); if (SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "exclusive").toBool()) { ao_defaults << "wasapi:exclusive=yes"; ao_defaults << "coreaudio:exclusive=yes"; } mpv::qt::set_option_variant(m_mpv, "ao-defaults", ao_defaults.join(',')); // set the audio device QVariant device = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "device"); mpv::qt::set_property_variant(m_mpv, "audio-device", device); // set the channel layout QVariant layout = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "channels"); mpv::qt::set_option_variant(m_mpv, "audio-channels", layout); QString resampleOpts = ""; bool normalize = SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "normalize").toBool(); resampleOpts += QString(":normalize=") + (normalize ? "yes" : "no"); mpv::qt::set_option_variant(m_mpv, "af-defaults", "lavrresample" + resampleOpts); QString passthroughCodecs; // passthrough doesn't make sense with basic type if (deviceType != AUDIO_DEVICE_TYPE_BASIC) { QStringList enabledCodecs; SettingsSection* audioSection = SettingsComponent::Get().getSection(SETTINGS_SECTION_AUDIO); QStringList codecs; if (deviceType == AUDIO_DEVICE_TYPE_SPDIF) codecs = AudioCodecsSPDIF(); else if (deviceType == AUDIO_DEVICE_TYPE_HDMI && audioSection->value("advanced").toBool()) codecs = AudioCodecsAll(); foreach (const QString& key, codecs) { if (audioSection->value("passthrough." + key).toBool()) enabledCodecs << key; } // dts-hd includes dts, but listing dts before dts-hd may disable dts-hd. if (enabledCodecs.indexOf("dts-hd") != -1) enabledCodecs.removeAll("dts"); passthroughCodecs = enabledCodecs.join(","); } mpv::qt::set_option_variant(m_mpv, "audio-spdif", passthroughCodecs); // if the user has indicated that we have a optical spdif connection // we need to set this extra option that allows us to transcode // 5.1 audio into a usable format, note that we only support AC3 // here for now. We might need to add support for DTS transcoding // if we see user requests for it. // bool doAc3Transcoding = false; if (deviceType == AUDIO_DEVICE_TYPE_SPDIF && SettingsComponent::Get().value(SETTINGS_SECTION_AUDIO, "passthrough.ac3").toBool()) { QLOG_INFO() << "Enabling audio AC3 transcoding (if needed)"; mpv::qt::set_option_variant(m_mpv, "af", "lavcac3enc"); doAc3Transcoding = true; } // Make a informational log message. QString audioConfig = QString(QString("Audio Config - device: %1, ") + "channel layout: %2, " + "passthrough codecs: %3, " + "ac3 transcoding: %4").arg(device.toString(), layout.toString(), passthroughCodecs.isEmpty() ? "none" : passthroughCodecs, doAc3Transcoding ? "yes" : "no"); QLOG_INFO() << qPrintable(audioConfig); } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::updateSubtitleSettings() { QVariant size = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "size"); mpv::qt::set_option_variant(m_mpv, "sub-text-font-size", size); QVariant colors_string = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "color"); auto colors = colors_string.toString().split(","); if (colors.length() == 2) { mpv::qt::set_option_variant(m_mpv, "sub-text-color", colors[0]); mpv::qt::set_option_variant(m_mpv, "sub-text-border-color", colors[1]); } QVariant subpos_string = SettingsComponent::Get().value(SETTINGS_SECTION_SUBTITLES, "placement"); auto subpos = subpos_string.toString().split(","); if (subpos.length() == 2) { mpv::qt::set_option_variant(m_mpv, "sub-text-align-x", subpos[0]); mpv::qt::set_option_variant(m_mpv, "sub-text-align-y", subpos[1]); } } /////////////////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::updateVideoSettings() { QVariant sync_mode = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "sync_mode"); mpv::qt::set_option_variant(m_mpv, "video-sync", sync_mode); QVariant hardware_decoding = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "hardware_decoding"); mpv::qt::set_property_variant(m_mpv, "hwdec", hardware_decoding.toBool() ? "auto" : "no"); QVariant deinterlace = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "deinterlace"); mpv::qt::set_option_variant(m_mpv, "deinterlace", deinterlace.toBool() ? "yes" : "no"); #ifndef TARGET_RPI double display_fps = DisplayComponent::Get().currentRefreshRate(); mpv::qt::set_option_variant(m_mpv, "display-fps", display_fps); #endif setAudioDelay(m_playbackAudioDelay); QVariant cache = SettingsComponent::Get().value(SETTINGS_SECTION_VIDEO, "cache"); mpv::qt::set_option_variant(m_mpv, "cache", cache.toInt() * 1024); } ///////////////////////////////////////////////////////////////////////////////////////// void PlayerComponent::userCommand(const QString& command) { QByteArray cmd_utf8 = command.toUtf8(); mpv_command_string(m_mpv, cmd_utf8.data()); } ///////////////////////////////////////////////////////////////////////////////////////// static QString get_mpv_osd(mpv_handle *ctx, const char *property) { char *s = mpv_get_property_osd_string(ctx, property); if (!s) return "-"; QString r = QString::fromUtf8(s); mpv_free(s); return r; } #define MPV_PROPERTY(p) get_mpv_osd(m_mpv, p) ///////////////////////////////////////////////////////////////////////////////////////// QString PlayerComponent::videoInformation() const { QString infoStr; QTextStream info(&infoStr); // check if video is playing if (mpv::qt::get_property_variant(m_mpv, "idle").toBool()) return ""; info << "File:" << endl; info << "URL: " << MPV_PROPERTY("path") << endl; info << "Container: " << MPV_PROPERTY("file-format") << endl; info << endl; info << "Video:" << endl; info << "Codec: " << MPV_PROPERTY("video-codec") << endl; info << "Size: " << MPV_PROPERTY("video-params/dw") << "x" << MPV_PROPERTY("video-params/dh") << endl; info << "FPS: " << MPV_PROPERTY("fps") << endl; info << "Aspect: " << MPV_PROPERTY("video-aspect") << endl; info << "Bitrate: " << MPV_PROPERTY("video-bitrate") << endl; info << "Display Sync: " << MPV_PROPERTY("display-sync-active") << endl; info << "Hardware Decoding: " << MPV_PROPERTY("hwdec-active") << " (" << MPV_PROPERTY("hwdec-detected") << ")" << endl; info << endl; info << "Audio: " << endl; info << "Codec: " << MPV_PROPERTY("audio-codec") << endl; info << "Bitrate: " << MPV_PROPERTY("audio-bitrate") << endl; info << "Channels (input): " << MPV_PROPERTY("audio-params/channels") << endl; info << "Channels (output): " << MPV_PROPERTY("audio-out-params/hr-channels") << " (" << MPV_PROPERTY("audio-out-params/channels") << ")" << endl; info << endl; info << "Performance: " << endl; info << "A/V: " << MPV_PROPERTY("avsync") << endl; info << "Change: " << MPV_PROPERTY("total-avsync-change") << endl; info << "Dropped frames: " << MPV_PROPERTY("vo-drop-frame-count") << endl; info << "Missed frames: " << MPV_PROPERTY("vo-missed-frame-count") << endl; info << endl; info << "Cache:" << endl; info << "Seconds: " << MPV_PROPERTY("demuxer-cache-duration") << endl; info << "Extra readahead: " << MPV_PROPERTY("cache-used") << endl; info << "Buffering: " << MPV_PROPERTY("cache-buffering-state") << endl; info << endl; info << "Misc: " << endl; info << "Time: " << MPV_PROPERTY("playback-time") << " / " << MPV_PROPERTY("duration") << " (" << MPV_PROPERTY("percent-pos") << "%)" << endl; info << "Pause state: " << MPV_PROPERTY("pause") << " / " << MPV_PROPERTY("paused-for-cache") << " / " << MPV_PROPERTY("core-idle") << endl; info << "Seeking: " << MPV_PROPERTY("seeking") << endl; info << "Seekable: " << MPV_PROPERTY("seekable") << " / " << MPV_PROPERTY("partially-seekable") << endl; info << flush; return infoStr; }