mirror of https://github.com/procxx/kepka.git
Show loaded till in playback slider.
This commit is contained in:
parent
0ce5405eef
commit
2255eb2c68
|
@ -119,6 +119,7 @@ struct TrackState {
|
|||
int64 receivedTill = 0;
|
||||
int64 length = 0;
|
||||
int frequency = kDefaultFrequency;
|
||||
int fileHeaderSize = 0;
|
||||
bool waitingForData = false;
|
||||
};
|
||||
|
||||
|
|
|
@ -63,6 +63,7 @@ struct AudioInformation {
|
|||
struct Information {
|
||||
VideoInformation video;
|
||||
AudioInformation audio;
|
||||
int headerSize = 0;
|
||||
};
|
||||
|
||||
template <typename Track>
|
||||
|
|
|
@ -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);
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -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();
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -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,
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -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);
|
||||
|
|
|
@ -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));
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -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;
|
||||
|
||||
|
|
Loading…
Reference in New Issue