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 receivedTill = 0;
|
||||||
int64 length = 0;
|
int64 length = 0;
|
||||||
int frequency = kDefaultFrequency;
|
int frequency = kDefaultFrequency;
|
||||||
|
int fileHeaderSize = 0;
|
||||||
bool waitingForData = false;
|
bool waitingForData = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue