Show loaded till in playback slider.

This commit is contained in:
John Preston 2019-05-31 14:45:35 +03:00
parent 0ce5405eef
commit 2255eb2c68
12 changed files with 78 additions and 33 deletions

View File

@ -119,6 +119,7 @@ struct TrackState {
int64 receivedTill = 0; int64 receivedTill = 0;
int64 length = 0; int64 length = 0;
int frequency = kDefaultFrequency; int frequency = kDefaultFrequency;
int fileHeaderSize = 0;
bool waitingForData = false; bool waitingForData = false;
}; };

View File

@ -63,6 +63,7 @@ struct AudioInformation {
struct Information { struct Information {
VideoInformation video; VideoInformation video;
AudioInformation audio; AudioInformation audio;
int headerSize = 0;
}; };
template <typename Track> template <typename Track>

View File

@ -251,7 +251,8 @@ void File::Context::start(crl::time position) {
return; return;
} }
if (!_delegate->fileReady(std::move(video), std::move(audio))) { const auto header = _reader->headerSize();
if (!_delegate->fileReady(header, std::move(video), std::move(audio))) {
return fail(Error::OpenFailed); return fail(Error::OpenFailed);
} }
_format = std::move(format); _format = std::move(format);

View File

@ -17,6 +17,7 @@ enum class Error;
class FileDelegate { class FileDelegate {
public: public:
[[nodiscard]] virtual bool fileReady( [[nodiscard]] virtual bool fileReady(
int headerSize,
Stream &&video, Stream &&video,
Stream &&audio) = 0; Stream &&audio) = 0;
virtual void fileError(Error error) = 0; virtual void fileError(Error error) = 0;

View File

@ -72,6 +72,9 @@ void SaveValidStartInformation(Information &to, Information &&from) {
if (from.video.state.duration != kTimeUnknown) { if (from.video.state.duration != kTimeUnknown) {
SaveValidVideoInformation(to.video, std::move(from.video)); SaveValidVideoInformation(to.video, std::move(from.video));
} }
if (from.headerSize && !to.headerSize) {
to.headerSize = from.headerSize;
}
} }
} // namespace } // namespace
@ -209,12 +212,13 @@ void Player::videoPlayedTill(crl::time position) {
trackPlayedTill(*_video, _information.video.state, position); trackPlayedTill(*_video, _information.video.state, position);
} }
bool Player::fileReady(Stream &&video, Stream &&audio) { bool Player::fileReady(int headerSize, Stream &&video, Stream &&audio) {
_waitingForData = false; _waitingForData = false;
const auto weak = base::make_weak(&_sessionGuard); const auto weak = base::make_weak(&_sessionGuard);
const auto ready = [=](const Information &data) { const auto ready = [=](const Information &data) {
crl::on_main(weak, [=, data = data]() mutable { crl::on_main(weak, [=, data = data]() mutable {
data.headerSize = headerSize;
streamReady(std::move(data)); streamReady(std::move(data));
}); });
}; };
@ -702,7 +706,9 @@ void Player::stop(bool stillActive) {
_durationByPackets = 0; _durationByPackets = 0;
_durationByLastAudioPacket = 0; _durationByLastAudioPacket = 0;
_durationByLastVideoPacket = 0; _durationByLastVideoPacket = 0;
const auto header = _information.headerSize;
_information = Information(); _information = Information();
_information.headerSize = header;
} }
std::optional<Error> Player::failed() const { std::optional<Error> Player::failed() const {
@ -801,6 +807,7 @@ Media::Player::TrackState Player::prepareLegacyState() const {
? getCurrentReceivedTill(result.length) ? getCurrentReceivedTill(result.length)
: 0; : 0;
result.frequency = kMsFrequency; result.frequency = kMsFrequency;
result.fileHeaderSize = _information.headerSize;
if (result.length == kTimeUnknown) { if (result.length == kTimeUnknown) {
const auto document = _options.audioId.audio(); const auto document = _options.audioId.audio();

View File

@ -79,7 +79,7 @@ private:
not_null<FileDelegate*> delegate(); not_null<FileDelegate*> delegate();
// FileDelegate methods are called only from the File thread. // FileDelegate methods are called only from the File thread.
bool fileReady(Stream &&video, Stream &&audio) override; bool fileReady(int headerSize, Stream &&video, Stream &&audio) override;
void fileError(Error error) override; void fileError(Error error) override;
void fileWaitingForData() override; void fileWaitingForData() override;
bool fileProcessPacket(Packet &&packet) override; bool fileProcessPacket(Packet &&packet) override;

View File

@ -381,6 +381,10 @@ void Reader::Slices::headerDone(bool fromCache) {
} }
} }
int Reader::Slices::headerSize() const {
return _header.parts.size() * kPartSize;
}
bool Reader::Slices::headerWontBeFilled() const { bool Reader::Slices::headerWontBeFilled() const {
return headerModeUnknown() return headerModeUnknown()
&& (_header.parts.size() >= kMaxPartsInHeader); && (_header.parts.size() >= kMaxPartsInHeader);
@ -1075,6 +1079,10 @@ void Reader::headerDone() {
_slices.headerDone(false); _slices.headerDone(false);
} }
int Reader::headerSize() const {
return _slices.headerSize();
}
bool Reader::fill( bool Reader::fill(
int offset, int offset,
bytes::span buffer, bytes::span buffer,

View File

@ -49,6 +49,7 @@ public:
not_null<crl::semaphore*> notify); not_null<crl::semaphore*> notify);
[[nodiscard]] std::optional<Error> streamingError() const; [[nodiscard]] std::optional<Error> streamingError() const;
void headerDone(); void headerDone();
[[nodiscard]] int headerSize() const;
// Thread safe. // Thread safe.
void startSleep(not_null<crl::semaphore*> wake); void startSleep(not_null<crl::semaphore*> wake);
@ -133,6 +134,7 @@ private:
Slices(int size, bool useCache); Slices(int size, bool useCache);
void headerDone(bool fromCache); void headerDone(bool fromCache);
[[nodiscard]] int headerSize() const;
[[nodiscard]] bool headerWontBeFilled() const; [[nodiscard]] bool headerWontBeFilled() const;
[[nodiscard]] bool headerModeUnknown() const; [[nodiscard]] bool headerModeUnknown() const;
[[nodiscard]] bool isFullInHeader() const; [[nodiscard]] bool isFullInHeader() const;

View File

@ -154,10 +154,22 @@ void PlaybackControls::fadeUpdated(float64 opacity) {
void PlaybackControls::updatePlayback(const Player::TrackState &state) { void PlaybackControls::updatePlayback(const Player::TrackState &state) {
updatePlayPauseResumeState(state); updatePlayPauseResumeState(state);
_playbackProgress->updateState(state); _playbackProgress->updateState(state, countDownloadedTillPercent(state));
updateTimeTexts(state); updateTimeTexts(state);
} }
float64 PlaybackControls::countDownloadedTillPercent(
const Player::TrackState &state) const {
if (_loadingReady > 0 && _loadingReady == _loadingTotal) {
return 1.;
}
const auto header = state.fileHeaderSize;
if (!header || _loadingReady <= header || _loadingTotal <= header) {
return 0.;
}
return (_loadingReady - header) / float64(_loadingTotal - header);
}
void PlaybackControls::setLoadingProgress(int ready, int total) { void PlaybackControls::setLoadingProgress(int ready, int total) {
if (_loadingReady == ready && _loadingTotal == total) { if (_loadingReady == ready && _loadingTotal == total) {
return; return;

View File

@ -64,6 +64,8 @@ private:
void fadeFinished(); void fadeFinished();
void fadeUpdated(float64 opacity); void fadeUpdated(float64 opacity);
void refreshFadeCache(); void refreshFadeCache();
[[nodiscard]] float64 countDownloadedTillPercent(
const Player::TrackState &state) const;
void updatePlayPauseResumeState(const Player::TrackState &state); void updatePlayPauseResumeState(const Player::TrackState &state);
void updateTimeTexts(const Player::TrackState &state); void updateTimeTexts(const Player::TrackState &state);

View File

@ -22,12 +22,14 @@ PlaybackProgress::PlaybackProgress()
: _valueAnimation([=](crl::time now) { : _valueAnimation([=](crl::time now) {
return valueAnimationCallback(now); return valueAnimationCallback(now);
}) })
, _receivedTillAnimation([=](crl::time now) { , _availableTillAnimation([=](crl::time now) {
return receivedTillAnimationCallback(now); return availableTillAnimationCallback(now);
}) { }) {
} }
void PlaybackProgress::updateState(const Player::TrackState &state) { void PlaybackProgress::updateState(
const Player::TrackState &state,
float64 loadedTillPercent) {
_playing = !Player::IsStopped(state.state); _playing = !Player::IsStopped(state.state);
const auto length = state.length; const auto length = state.length;
const auto position = Player::IsStoppedAtEnd(state.state) const auto position = Player::IsStoppedAtEnd(state.state)
@ -38,6 +40,12 @@ void PlaybackProgress::updateState(const Player::TrackState &state) {
const auto receivedTill = (length && state.receivedTill > position) const auto receivedTill = (length && state.receivedTill > position)
? state.receivedTill ? state.receivedTill
: -1; : -1;
const auto loadedTill = (loadedTillPercent != 0.)
? int64(std::floor(loadedTillPercent * length))
: -1;
const auto availableTill = (length && loadedTill > position)
? std::max(receivedTill, loadedTill)
: receivedTill;
const auto wasInLoadingState = _inLoadingState; const auto wasInLoadingState = _inLoadingState;
if (wasInLoadingState) { if (wasInLoadingState) {
@ -52,8 +60,8 @@ void PlaybackProgress::updateState(const Player::TrackState &state) {
: length : length
? snap(float64(position) / length, 0., 1.) ? snap(float64(position) / length, 0., 1.)
: 0.; : 0.;
const auto receivedTillProgress = (receivedTill > position) const auto availableTillProgress = (availableTill > position)
? snap(float64(receivedTill) / length, 0., 1.) ? snap(float64(availableTill) / length, 0., 1.)
: -1.; : -1.;
const auto animatedPosition = position + (state.frequency * kPlaybackAnimationDurationMs / 1000); const auto animatedPosition = position + (state.frequency * kPlaybackAnimationDurationMs / 1000);
const auto animatedProgress = length ? qMax(float64(animatedPosition) / length, 0.) : 0.; const auto animatedProgress = length ? qMax(float64(animatedPosition) / length, 0.) : 0.;
@ -66,9 +74,9 @@ void PlaybackProgress::updateState(const Player::TrackState &state) {
_position = position; _position = position;
_length = length; _length = length;
} }
if (receivedTill != _receivedTill) { if (availableTill != _availableTill) {
setReceivedTill(receivedTillProgress); setAvailableTill(availableTillProgress);
_receivedTill = receivedTill; _availableTill = availableTill;
} }
} }
@ -99,18 +107,18 @@ void PlaybackProgress::setValue(float64 value, bool animated) {
emitUpdatedValue(); emitUpdatedValue();
} }
void PlaybackProgress::setReceivedTill(float64 value) { void PlaybackProgress::setAvailableTill(float64 value) {
const auto current = a_receivedTill.current(); const auto current = a_availableTill.current();
if (value > current && current > 0.) { if (value > current && current > 0.) {
receivedTillAnimationCallback(crl::now()); availableTillAnimationCallback(crl::now());
a_receivedTill.start(value); a_availableTill.start(value);
_receivedTillAnimation.start(); _availableTillAnimation.start();
} else if (value > a_value.current()) { } else if (value > a_value.current()) {
a_receivedTill = anim::value(a_value.current(), value); a_availableTill = anim::value(a_value.current(), value);
_receivedTillAnimation.start(); _availableTillAnimation.start();
} else { } else {
a_receivedTill = anim::value(-1., -1.); a_availableTill = anim::value(-1., -1.);
_receivedTillAnimation.stop(); _availableTillAnimation.stop();
} }
emitUpdatedValue(); emitUpdatedValue();
} }
@ -129,15 +137,15 @@ bool PlaybackProgress::valueAnimationCallback(float64 now) {
return (dt < 1.); return (dt < 1.);
} }
bool PlaybackProgress::receivedTillAnimationCallback(float64 now) { bool PlaybackProgress::availableTillAnimationCallback(float64 now) {
const auto time = now - _receivedTillAnimation.started(); const auto time = now - _availableTillAnimation.started();
const auto dt = anim::Disabled() const auto dt = anim::Disabled()
? 1. ? 1.
: (time / kPlaybackAnimationDurationMs); : (time / kPlaybackAnimationDurationMs);
if (dt >= 1.) { if (dt >= 1.) {
a_receivedTill.finish(); a_availableTill.finish();
} else { } else {
a_receivedTill.update(dt, anim::linear); a_availableTill.update(dt, anim::linear);
} }
emitUpdatedValue(); emitUpdatedValue();
return (dt < 1.); return (dt < 1.);
@ -146,8 +154,8 @@ bool PlaybackProgress::receivedTillAnimationCallback(float64 now) {
void PlaybackProgress::emitUpdatedValue() { void PlaybackProgress::emitUpdatedValue() {
if (_valueChanged) { if (_valueChanged) {
const auto value = a_value.current(); const auto value = a_value.current();
const auto receivedTill = a_receivedTill.current(); const auto availableTill = a_availableTill.current();
_valueChanged(value, std::max(value, receivedTill)); _valueChanged(value, std::max(value, availableTill));
} }
} }

View File

@ -29,20 +29,22 @@ public:
void setValue(float64 value, bool animated); void setValue(float64 value, bool animated);
float64 value() const; float64 value() const;
void updateState(const Player::TrackState &state); void updateState(
const Player::TrackState &state,
float64 loadedTillPercent = 0.);
void updateLoadingState(float64 progress); void updateLoadingState(float64 progress);
private: private:
bool valueAnimationCallback(float64 now); bool valueAnimationCallback(float64 now);
bool receivedTillAnimationCallback(float64 now); bool availableTillAnimationCallback(float64 now);
void setReceivedTill(float64 value); void setAvailableTill(float64 value);
void emitUpdatedValue(); void emitUpdatedValue();
// This can animate for a very long time (like in music playing), // This can animate for a very long time (like in music playing),
// so it should be a Basic, not a Simple animation, because // so it should be a Basic, not a Simple animation, because
// Simple-s pauses mtproto responses/updates handling while playing. // Simple-s pauses mtproto responses/updates handling while playing.
anim::value a_value, a_receivedTill; anim::value a_value, a_availableTill;
Ui::Animations::Basic _valueAnimation, _receivedTillAnimation; Ui::Animations::Basic _valueAnimation, _availableTillAnimation;
Fn<void(float64,float64)> _valueChanged; Fn<void(float64,float64)> _valueChanged;
bool _inLoadingState = false; bool _inLoadingState = false;
@ -50,7 +52,7 @@ private:
int64 _position = 0; int64 _position = 0;
int64 _length = 0; int64 _length = 0;
int64 _receivedTill = -1; int64 _availableTill = -1;
bool _playing = false; bool _playing = false;