tdesktop/Telegram/SourceFiles/media/player/media_player_instance.cpp

750 lines
19 KiB
C++
Raw Normal View History

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "media/player/media_player_instance.h"
#include "data/data_document.h"
#include "data/data_session.h"
#include "media/audio/media_audio.h"
#include "media/audio/media_audio_capture.h"
2019-12-11 10:15:48 +00:00
#include "media/streaming/media_streaming_document.h"
#include "media/streaming/media_streaming_reader.h"
#include "media/view/media_view_playback_progress.h"
#include "calls/calls_instance.h"
#include "history/history.h"
#include "history/history_item.h"
#include "data/data_media_types.h"
#include "data/data_file_origin.h"
#include "window/window_session_controller.h"
2018-11-16 12:15:14 +00:00
#include "core/shortcuts.h"
#include "core/application.h"
2019-06-06 09:37:12 +00:00
#include "main/main_account.h" // Account::sessionValue.
#include "mainwindow.h"
2019-07-24 11:45:24 +00:00
#include "main/main_session.h"
#include "facades.h"
namespace Media {
namespace Player {
namespace {
Instance *SingleInstance = nullptr;
constexpr auto kVoicePlaybackSpeedMultiplier = 1.7;
// Preload X message ids before and after current.
constexpr auto kIdsLimit = 32;
// Preload next messages if we went further from current than that.
constexpr auto kIdsPreloadAfter = 28;
} // namespace
void start(not_null<Audio::Instance*> instance) {
Audio::Start(instance);
Capture::Start();
SingleInstance = new Instance();
}
void finish(not_null<Audio::Instance*> instance) {
delete base::take(SingleInstance);
Capture::Finish();
Audio::Finish(instance);
}
2019-02-28 21:03:25 +00:00
struct Instance::Streamed {
Streamed(
AudioMsgId id,
2019-12-11 10:15:48 +00:00
std::shared_ptr<Streaming::Document> document);
2019-02-28 21:03:25 +00:00
AudioMsgId id;
2019-12-11 10:15:48 +00:00
std::shared_ptr<Streaming::Document> shared;
View::PlaybackProgress progress;
bool clearing = false;
2019-12-11 10:15:48 +00:00
rpl::lifetime lifetime;
2019-02-28 21:03:25 +00:00
};
Instance::Streamed::Streamed(
AudioMsgId id,
2019-12-11 10:15:48 +00:00
std::shared_ptr<Streaming::Document> document)
2019-02-28 21:03:25 +00:00
: id(id)
2019-12-11 10:15:48 +00:00
, shared(std::move(document)) {
2019-02-28 21:03:25 +00:00
}
Instance::Data::Data(AudioMsgId::Type type, SharedMediaType overview)
: type(type)
, overview(overview) {
}
Instance::Data::Data(Data &&other) = default;
Instance::Data &Instance::Data::operator=(Data &&other) = default;
Instance::Data::~Data() = default;
Instance::Instance()
2017-12-08 18:27:28 +00:00
: _songData(AudioMsgId::Type::Song, SharedMediaType::MusicFile)
, _voiceData(AudioMsgId::Type::Voice, SharedMediaType::RoundVoiceFile) {
subscribe(Media::Player::Updated(), [this](const AudioMsgId &audioId) {
handleSongUpdate(audioId);
});
2019-07-24 11:45:24 +00:00
// While we have one Media::Player::Instance for all sessions we have to do this.
2019-06-06 09:37:12 +00:00
Core::App().activeAccount().sessionValue(
2019-07-24 11:45:24 +00:00
) | rpl::start_with_next([=](Main::Session *session) {
2019-06-06 09:37:12 +00:00
if (session) {
subscribe(session->calls().currentCallChanged(), [=](Calls::Call *call) {
if (call) {
2018-10-31 06:51:19 +00:00
pauseOnCall(AudioMsgId::Type::Voice);
pauseOnCall(AudioMsgId::Type::Song);
} else {
resumeOnCall(AudioMsgId::Type::Voice);
resumeOnCall(AudioMsgId::Type::Song);
}
});
2019-06-06 09:37:12 +00:00
} else {
const auto reset = [&](AudioMsgId::Type type) {
const auto data = getData(type);
*data = Data(type, data->overview);
};
reset(AudioMsgId::Type::Voice);
reset(AudioMsgId::Type::Song);
}
2019-06-06 09:37:12 +00:00
}, _lifetime);
2018-11-16 12:15:14 +00:00
setupShortcuts();
}
2019-02-28 21:03:25 +00:00
Instance::~Instance() = default;
AudioMsgId::Type Instance::getActiveType() const {
if (const auto data = getData(AudioMsgId::Type::Voice)) {
if (data->current) {
const auto state = getState(data->type);
if (!IsStoppedOrStopping(state.state)) {
return data->type;
}
}
}
return AudioMsgId::Type::Song;
}
void Instance::handleSongUpdate(const AudioMsgId &audioId) {
emitUpdate(audioId.type(), [&](const AudioMsgId &playing) {
return (audioId == playing);
});
}
void Instance::setCurrent(const AudioMsgId &audioId) {
2019-02-28 21:03:25 +00:00
if (const auto data = getData(audioId.type())) {
if (data->current == audioId) {
return;
}
2019-03-05 13:56:27 +00:00
const auto changed = [&](const AudioMsgId & check) {
return (check.audio() != audioId.audio())
|| (check.contextId() != audioId.contextId());
};
if (changed(data->current)
&& data->streamed
&& changed(data->streamed->id)) {
2019-03-05 13:56:27 +00:00
clearStreamed(data);
}
2019-02-28 21:03:25 +00:00
data->current = audioId;
data->isPlaying = false;
2019-04-25 12:45:15 +00:00
const auto item = Auth().data().message(data->current.contextId());
2019-02-28 21:03:25 +00:00
if (item) {
data->history = item->history()->migrateToOrMe();
data->migrated = data->history->migrateFrom();
} else {
data->history = nullptr;
data->migrated = nullptr;
}
_trackChangedNotifier.notify(data->type, true);
refreshPlaylist(data);
}
}
2019-03-05 13:56:27 +00:00
void Instance::clearStreamed(not_null<Data*> data) {
if (!data->streamed || data->streamed->clearing) {
2019-03-05 13:56:27 +00:00
return;
}
data->streamed->clearing = true;
2019-12-11 10:15:48 +00:00
data->streamed->shared->stop();
2019-03-05 13:56:27 +00:00
data->isPlaying = false;
requestRoundVideoResize();
2019-03-05 13:56:27 +00:00
emitUpdate(data->type);
data->streamed = nullptr;
App::wnd()->sessionController()->disableGifPauseReason(
Window::GifPauseReason::RoundPlaying);
2019-03-05 13:56:27 +00:00
}
void Instance::refreshPlaylist(not_null<Data*> data) {
if (!validPlaylist(data)) {
validatePlaylist(data);
}
playlistUpdated(data);
}
void Instance::playlistUpdated(not_null<Data*> data) {
if (data->playlistSlice) {
const auto fullId = data->current.contextId();
data->playlistIndex = data->playlistSlice->indexOf(fullId);
} else {
2018-09-21 16:28:46 +00:00
data->playlistIndex = std::nullopt;
}
data->playlistChanges.fire({});
}
bool Instance::validPlaylist(not_null<Data*> data) {
if (const auto key = playlistKey(data)) {
if (!data->playlistSlice) {
return false;
}
using Key = SliceKey;
const auto inSameDomain = [](const Key &a, const Key &b) {
return (a.peerId == b.peerId)
&& (a.migratedPeerId == b.migratedPeerId);
};
const auto countDistanceInData = [&](const Key &a, const Key &b) {
return [&](const SparseIdsMergedSlice &data) {
return inSameDomain(a, b)
? data.distance(a, b)
2018-09-21 16:28:46 +00:00
: std::optional<int>();
};
};
if (key == data->playlistRequestedKey) {
return true;
} else if (!data->playlistSliceKey
|| !data->playlistRequestedKey
|| *data->playlistRequestedKey != *data->playlistSliceKey) {
return false;
}
auto distance = data->playlistSlice
| countDistanceInData(*key, *data->playlistRequestedKey)
| func::abs;
if (distance) {
return (*distance < kIdsPreloadAfter);
}
}
return !data->playlistSlice;
}
void Instance::validatePlaylist(not_null<Data*> data) {
if (const auto key = playlistKey(data)) {
data->playlistRequestedKey = key;
SharedMediaMergedViewer(
SharedMediaMergedKey(*key, data->overview),
kIdsLimit,
kIdsLimit
) | rpl::start_with_next([=](SparseIdsMergedSlice &&update) {
data->playlistSlice = std::move(update);
data->playlistSliceKey = key;
playlistUpdated(data);
}, data->playlistLifetime);
} else {
2018-09-21 16:28:46 +00:00
data->playlistSlice = std::nullopt;
data->playlistSliceKey = data->playlistRequestedKey = std::nullopt;
playlistUpdated(data);
}
}
auto Instance::playlistKey(not_null<Data*> data) const
2018-09-21 16:28:46 +00:00
-> std::optional<SliceKey> {
const auto contextId = data->current.contextId();
const auto history = data->history;
if (!contextId || !history || !IsServerMsgId(contextId.msg)) {
return {};
}
const auto universalId = (contextId.channel == history->channelId())
? contextId.msg
: (contextId.msg - ServerMaxMsgId);
return SliceKey(
data->history->peer->id,
data->migrated ? data->migrated->peer->id : 0,
universalId);
}
HistoryItem *Instance::itemByIndex(not_null<Data*> data, int index) {
if (!data->playlistSlice
|| index < 0
|| index >= data->playlistSlice->size()) {
return nullptr;
}
const auto fullId = (*data->playlistSlice)[index];
2019-04-25 12:45:15 +00:00
return Auth().data().message(fullId);
}
2017-12-08 18:27:28 +00:00
bool Instance::moveInPlaylist(
not_null<Data*> data,
int delta,
bool autonext) {
if (!data->playlistIndex) {
return false;
}
const auto newIndex = *data->playlistIndex + delta;
if (const auto item = itemByIndex(data, newIndex)) {
if (const auto media = item->media()) {
if (const auto document = media->document()) {
if (autonext) {
_switchToNextNotifier.notify({
data->current,
item->fullId()
});
}
if (document->isAudioFile()
|| document->isVoiceMessage()
|| document->isVideoMessage()) {
play(AudioMsgId(document, item->fullId()));
}
return true;
}
}
}
2017-12-08 18:27:28 +00:00
return false;
}
2017-12-08 18:27:28 +00:00
bool Instance::previousAvailable(AudioMsgId::Type type) const {
const auto data = getData(type);
Assert(data != nullptr);
return data->playlistIndex
&& data->playlistSlice
&& (*data->playlistIndex > 0);
2017-12-08 18:27:28 +00:00
}
2017-12-08 18:27:28 +00:00
bool Instance::nextAvailable(AudioMsgId::Type type) const {
const auto data = getData(type);
Assert(data != nullptr);
return data->playlistIndex
&& data->playlistSlice
&& (*data->playlistIndex + 1 < data->playlistSlice->size());
}
2017-12-08 18:27:28 +00:00
rpl::producer<> Media::Player::Instance::playlistChanges(
AudioMsgId::Type type) const {
const auto data = getData(type);
Assert(data != nullptr);
return data->playlistChanges.events();
2017-12-08 18:27:28 +00:00
}
Instance *instance() {
Expects(SingleInstance != nullptr);
return SingleInstance;
}
void Instance::play(AudioMsgId::Type type) {
2019-02-28 21:03:25 +00:00
if (const auto data = getData(type)) {
if (!data->streamed || IsStopped(getState(type).state)) {
play(data->current);
} else {
2019-12-11 10:15:48 +00:00
if (data->streamed->shared->active()) {
data->streamed->shared->resume();
}
emitUpdate(type);
}
2018-10-31 06:51:19 +00:00
data->resumeOnCallEnd = false;
}
}
void Instance::play(const AudioMsgId &audioId) {
const auto document = audioId.audio();
if (!document) {
return;
}
if (document->isAudioFile()
|| document->isVoiceMessage()
|| document->isVideoMessage()) {
2019-12-11 10:15:48 +00:00
auto shared = document->owner().documentStreamer(
document,
audioId.contextId());
2019-12-11 10:15:48 +00:00
if (!shared) {
2019-02-28 21:03:25 +00:00
return;
}
2019-12-11 10:15:48 +00:00
playStreamed(audioId, std::move(shared));
}
if (document->isVoiceMessage() || document->isVideoMessage()) {
document->owner().markMediaRead(document);
}
}
2019-02-28 21:03:25 +00:00
void Instance::playPause(const AudioMsgId &audioId) {
const auto now = current(audioId.type());
if (now.audio() == audioId.audio()
&& now.contextId() == audioId.contextId()) {
playPause(audioId.type());
} else {
play(audioId);
}
}
void Instance::playStreamed(
const AudioMsgId &audioId,
2019-12-11 10:15:48 +00:00
std::shared_ptr<Streaming::Document> shared) {
2019-02-28 21:03:25 +00:00
Expects(audioId.audio() != nullptr);
const auto data = getData(audioId.type());
Assert(data != nullptr);
clearStreamed(data);
2019-02-28 21:03:25 +00:00
data->streamed = std::make_unique<Streamed>(
audioId,
2019-12-11 10:15:48 +00:00
std::move(shared));
2019-02-28 21:03:25 +00:00
2019-12-11 10:15:48 +00:00
data->streamed->shared->player().updates(
2019-02-28 21:03:25 +00:00
) | rpl::start_with_next_error([=](Streaming::Update &&update) {
handleStreamingUpdate(data, std::move(update));
}, [=](Streaming::Error &&error) {
2019-02-28 21:03:25 +00:00
handleStreamingError(data, std::move(error));
2019-12-11 10:15:48 +00:00
}, data->streamed->lifetime);
2019-02-28 21:03:25 +00:00
2019-12-11 10:15:48 +00:00
data->streamed->shared->play(streamingOptions(audioId));
2019-02-28 21:03:25 +00:00
emitUpdate(audioId.type());
}
Streaming::PlaybackOptions Instance::streamingOptions(
const AudioMsgId &audioId,
crl::time position) {
const auto document = audioId.audio();
2019-02-28 21:03:25 +00:00
auto result = Streaming::PlaybackOptions();
result.mode = (document && document->isVideoMessage())
? Streaming::Mode::Both
: Streaming::Mode::Audio;
result.speed = (document
&& (document->isVoiceMessage() || document->isVideoMessage())
&& Global::VoiceMsgPlaybackDoubled())
? kVoicePlaybackSpeedMultiplier
: 1.;
2019-02-28 21:03:25 +00:00
result.audioId = audioId;
result.position = position;
return result;
}
void Instance::pause(AudioMsgId::Type type) {
2019-02-28 21:03:25 +00:00
if (const auto data = getData(type)) {
if (data->streamed) {
2019-12-11 10:15:48 +00:00
if (data->streamed->shared->active()) {
data->streamed->shared->pause();
}
2019-02-28 21:03:25 +00:00
emitUpdate(type);
}
}
}
void Instance::stop(AudioMsgId::Type type) {
2018-10-31 06:51:19 +00:00
if (const auto data = getData(type)) {
2019-02-28 21:03:25 +00:00
if (data->streamed) {
clearStreamed(data);
2019-02-28 21:03:25 +00:00
}
2018-10-31 06:51:19 +00:00
data->resumeOnCallEnd = false;
}
}
void Instance::playPause(AudioMsgId::Type type) {
2019-02-28 21:03:25 +00:00
if (const auto data = getData(type)) {
if (!data->streamed) {
play(data->current);
} else {
2019-12-11 10:15:48 +00:00
const auto shared = data->streamed->shared.get();
if (!shared->active()) {
shared->play(streamingOptions(data->streamed->id));
} else if (shared->paused()) {
shared->resume();
2019-02-28 21:03:25 +00:00
} else {
2019-12-11 10:15:48 +00:00
shared->pause();
2019-02-28 21:03:25 +00:00
}
emitUpdate(type);
}
2018-10-31 06:51:19 +00:00
data->resumeOnCallEnd = false;
}
}
void Instance::pauseOnCall(AudioMsgId::Type type) {
2019-02-28 21:03:25 +00:00
const auto state = getState(type);
2018-10-31 06:51:19 +00:00
if (!state.id
|| IsStopped(state.state)
|| IsPaused(state.state)
|| state.state == State::Pausing) {
return;
}
pause(type);
if (const auto data = getData(type)) {
data->resumeOnCallEnd = true;
}
}
void Instance::resumeOnCall(AudioMsgId::Type type) {
if (const auto data = getData(type)) {
if (data->resumeOnCallEnd) {
data->resumeOnCallEnd = false;
2018-11-09 07:26:33 +00:00
play(type);
2018-10-31 06:51:19 +00:00
}
}
}
bool Instance::next(AudioMsgId::Type type) {
2018-10-31 06:51:19 +00:00
if (const auto data = getData(type)) {
return moveInPlaylist(data, 1, false);
}
return false;
}
bool Instance::previous(AudioMsgId::Type type) {
2018-10-31 06:51:19 +00:00
if (const auto data = getData(type)) {
return moveInPlaylist(data, -1, false);
}
return false;
}
void Instance::playPauseCancelClicked(AudioMsgId::Type type) {
if (isSeeking(type)) {
return;
}
2019-02-28 21:03:25 +00:00
const auto data = getData(type);
if (!data) {
return;
}
const auto state = getState(type);
const auto stopped = IsStoppedOrStopping(state.state);
const auto showPause = ShowPauseIcon(state.state);
const auto audio = state.id.audio();
if (audio && audio->loading() && !data->streamed) {
audio->cancel();
} else if (showPause) {
pause(type);
} else {
play(type);
}
}
void Instance::startSeeking(AudioMsgId::Type type) {
if (auto data = getData(type)) {
data->seeking = data->current;
}
pause(type);
2019-02-28 21:03:25 +00:00
emitUpdate(type);
}
2019-02-28 21:03:25 +00:00
void Instance::finishSeeking(AudioMsgId::Type type, float64 progress) {
if (const auto data = getData(type)) {
2019-12-11 10:15:48 +00:00
if (const auto streamed = data->streamed.get()) {
const auto &info = streamed->shared->info();
const auto duration = info.audio.state.duration;
2019-02-28 21:03:25 +00:00
if (duration != kTimeUnknown) {
const auto position = crl::time(std::round(
std::clamp(progress, 0., 1.) * duration));
2019-12-11 10:15:48 +00:00
streamed->shared->play(streamingOptions(
streamed->id,
2019-02-28 21:03:25 +00:00
position));
emitUpdate(type);
}
}
}
cancelSeeking(type);
}
void Instance::cancelSeeking(AudioMsgId::Type type) {
if (const auto data = getData(type)) {
data->seeking = AudioMsgId();
}
2019-02-28 21:03:25 +00:00
emitUpdate(type);
}
void Instance::updateVoicePlaybackSpeed() {
if (const auto data = getData(AudioMsgId::Type::Voice)) {
if (const auto streamed = data->streamed.get()) {
2019-12-11 10:15:48 +00:00
streamed->shared->setSpeed(Global::VoiceMsgPlaybackDoubled()
? kVoicePlaybackSpeedMultiplier
: 1.);
}
}
}
void Instance::documentLoadProgress(DocumentData *document) {
const auto type = document->isAudioFile()
? AudioMsgId::Type::Song
: AudioMsgId::Type::Voice;
2019-02-28 21:03:25 +00:00
emitUpdate(type, [&](const AudioMsgId &audioId) {
return (audioId.audio() == document);
});
}
2019-02-28 21:03:25 +00:00
void Instance::emitUpdate(AudioMsgId::Type type) {
emitUpdate(type, [](const AudioMsgId &playing) { return true; });
}
2019-02-28 21:03:25 +00:00
TrackState Instance::getState(AudioMsgId::Type type) const {
if (const auto data = getData(type)) {
if (data->streamed) {
2019-12-11 10:15:48 +00:00
return data->streamed->shared->player().prepareLegacyState();
2019-02-28 21:03:25 +00:00
}
}
return TrackState();
}
2019-12-11 10:15:48 +00:00
Streaming::Document *Instance::roundVideoStreamed(HistoryItem *item) const {
if (!item) {
return nullptr;
} else if (const auto data = getData(AudioMsgId::Type::Voice)) {
if (const auto streamed = data->streamed.get()) {
if (streamed->id.contextId() == item->fullId()) {
2019-12-11 10:15:48 +00:00
const auto player = &streamed->shared->player();
if (player->ready() && !player->videoSize().isEmpty()) {
2019-12-11 10:15:48 +00:00
return streamed->shared.get();
}
}
}
}
return nullptr;
}
View::PlaybackProgress *Instance::roundVideoPlayback(
HistoryItem *item) const {
2019-12-11 10:15:48 +00:00
return roundVideoStreamed(item)
? &getData(AudioMsgId::Type::Voice)->streamed->progress
: nullptr;
2019-02-28 21:03:25 +00:00
}
2019-02-28 21:03:25 +00:00
template <typename CheckCallback>
void Instance::emitUpdate(AudioMsgId::Type type, CheckCallback check) {
if (const auto data = getData(type)) {
const auto state = getState(type);
if (!state.id || !check(state.id)) {
return;
}
setCurrent(state.id);
2019-12-11 10:15:48 +00:00
if (const auto streamed = data->streamed.get()) {
if (!streamed->shared->info().video.size.isEmpty()) {
streamed->progress.updateState(state);
}
}
_updatedNotifier.fire_copy({state});
if (data->isPlaying && state.state == State::StoppedAtEnd) {
if (data->repeatEnabled) {
play(data->current);
} else if (!moveInPlaylist(data, 1, true)) {
_tracksFinishedNotifier.notify(type);
}
}
2019-03-05 13:56:27 +00:00
data->isPlaying = !IsStopped(state.state);
}
}
2018-11-16 12:15:14 +00:00
void Instance::setupShortcuts() {
Shortcuts::Requests(
) | rpl::start_with_next([=](not_null<Shortcuts::Request*> request) {
using Command = Shortcuts::Command;
request->check(Command::MediaPlay) && request->handle([=] {
play();
return true;
});
request->check(Command::MediaPause) && request->handle([=] {
pause();
return true;
});
request->check(Command::MediaPlayPause) && request->handle([=] {
playPause();
return true;
});
request->check(Command::MediaStop) && request->handle([=] {
stop();
return true;
});
request->check(Command::MediaPrevious) && request->handle([=] {
previous();
return true;
});
request->check(Command::MediaNext) && request->handle([=] {
next();
return true;
});
}, _lifetime);
}
2019-02-28 21:03:25 +00:00
void Instance::handleStreamingUpdate(
not_null<Data*> data,
Streaming::Update &&update) {
using namespace Streaming;
update.data.match([&](Information &update) {
2019-12-11 10:15:48 +00:00
if (!update.video.size.isEmpty()) {
data->streamed->progress.setValueChangedCallback([=](
float64,
float64) {
requestRoundVideoRepaint();
});
App::wnd()->sessionController()->enableGifPauseReason(
Window::GifPauseReason::RoundPlaying);
requestRoundVideoResize();
}
2019-02-28 21:03:25 +00:00
emitUpdate(data->type);
}, [&](PreloadedVideo &update) {
//emitUpdate(data->type, [](AudioMsgId) { return true; });
2019-02-28 21:03:25 +00:00
}, [&](UpdateVideo &update) {
emitUpdate(data->type);
}, [&](PreloadedAudio &update) {
2019-02-28 21:03:25 +00:00
//emitUpdate(data->type, [](AudioMsgId) { return true; });
}, [&](UpdateAudio &update) {
emitUpdate(data->type);
}, [&](WaitingForData) {
}, [&](MutedByOther) {
}, [&](Finished) {
emitUpdate(data->type);
2019-12-11 10:15:48 +00:00
if (data->streamed && data->streamed->shared->player().finished()) {
clearStreamed(data);
}
2019-02-28 21:03:25 +00:00
});
}
HistoryItem *Instance::roundVideoItem() const {
const auto data = getData(AudioMsgId::Type::Voice);
return (data->streamed
2019-12-11 10:15:48 +00:00
&& !data->streamed->shared->info().video.size.isEmpty())
2019-04-25 12:45:15 +00:00
? Auth().data().message(data->streamed->id.contextId())
: nullptr;
}
void Instance::requestRoundVideoResize() const {
if (const auto item = roundVideoItem()) {
Auth().data().requestItemResize(item);
}
}
void Instance::requestRoundVideoRepaint() const {
if (const auto item = roundVideoItem()) {
Auth().data().requestItemRepaint(item);
}
}
2019-02-28 21:03:25 +00:00
void Instance::handleStreamingError(
not_null<Data*> data,
Streaming::Error &&error) {
2019-03-05 13:56:27 +00:00
Expects(data->streamed != nullptr);
const auto document = data->streamed->id.audio();
const auto contextId = data->streamed->id.contextId();
if (error == Streaming::Error::NotStreamable) {
DocumentSaveClickHandler::Save(
(contextId ? contextId : ::Data::FileOrigin()),
document);
2019-03-05 13:56:27 +00:00
} else if (error == Streaming::Error::OpenFailed) {
DocumentSaveClickHandler::Save(
(contextId ? contextId : ::Data::FileOrigin()),
document,
DocumentSaveClickHandler::Mode::ToFile);
2019-03-05 13:56:27 +00:00
}
emitUpdate(data->type);
2019-12-11 10:15:48 +00:00
if (data->streamed && data->streamed->shared->player().failed()) {
clearStreamed(data);
}
2019-02-28 21:03:25 +00:00
}
} // namespace Player
} // namespace Media