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 length = 0;
int frequency = kDefaultFrequency;
int fileHeaderSize = 0;
bool waitingForData = false;
};

View File

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

View File

@ -251,7 +251,8 @@ void File::Context::start(crl::time position) {
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);
}
_format = std::move(format);

View File

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

View File

@ -79,7 +79,7 @@ private:
not_null<FileDelegate*> delegate();
// 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 fileWaitingForData() 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 {
return headerModeUnknown()
&& (_header.parts.size() >= kMaxPartsInHeader);
@ -1075,6 +1079,10 @@ void Reader::headerDone() {
_slices.headerDone(false);
}
int Reader::headerSize() const {
return _slices.headerSize();
}
bool Reader::fill(
int offset,
bytes::span buffer,

View File

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

View File

@ -154,10 +154,22 @@ void PlaybackControls::fadeUpdated(float64 opacity) {
void PlaybackControls::updatePlayback(const Player::TrackState &state) {
updatePlayPauseResumeState(state);
_playbackProgress->updateState(state);
_playbackProgress->updateState(state, countDownloadedTillPercent(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) {
if (_loadingReady == ready && _loadingTotal == total) {
return;

View File

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

View File

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

View File

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