mirror of https://github.com/procxx/kepka.git
Display receivedTill in video player controls.
This commit is contained in:
parent
e2eb9cea00
commit
71b733a018
|
@ -116,6 +116,7 @@ struct TrackState {
|
||||||
AudioMsgId id;
|
AudioMsgId id;
|
||||||
State state = State::Stopped;
|
State state = State::Stopped;
|
||||||
int64 position = 0;
|
int64 position = 0;
|
||||||
|
int64 receivedTill = 0;
|
||||||
int64 length = 0;
|
int64 length = 0;
|
||||||
int frequency = kDefaultFrequency;
|
int frequency = kDefaultFrequency;
|
||||||
bool waitingForData = false;
|
bool waitingForData = false;
|
||||||
|
|
|
@ -57,7 +57,7 @@ RoundController::RoundController(
|
||||||
[=](Clip::Notification notification) { callback(notification); },
|
[=](Clip::Notification notification) { callback(notification); },
|
||||||
Clip::Reader::Mode::Video);
|
Clip::Reader::Mode::Video);
|
||||||
_playbackProgress = std::make_unique<View::PlaybackProgress>();
|
_playbackProgress = std::make_unique<View::PlaybackProgress>();
|
||||||
_playbackProgress->setValueChangedCallback([=](float64 value) {
|
_playbackProgress->setValueChangedCallback([=](float64, float64) {
|
||||||
Auth().data().requestItemRepaint(_context);
|
Auth().data().requestItemRepaint(_context);
|
||||||
});
|
});
|
||||||
Auth().data().markMediaRead(_data);
|
Auth().data().markMediaRead(_data);
|
||||||
|
|
|
@ -97,7 +97,7 @@ Widget::Widget(QWidget *parent) : RpWidget(parent)
|
||||||
_playbackProgress->setInLoadingStateChangedCallback([this](bool loading) {
|
_playbackProgress->setInLoadingStateChangedCallback([this](bool loading) {
|
||||||
_playbackSlider->setDisabled(loading);
|
_playbackSlider->setDisabled(loading);
|
||||||
});
|
});
|
||||||
_playbackProgress->setValueChangedCallback([this](float64 value) {
|
_playbackProgress->setValueChangedCallback([this](float64 value, float64) {
|
||||||
_playbackSlider->setValue(value);
|
_playbackSlider->setValue(value);
|
||||||
});
|
});
|
||||||
_playbackSlider->setChangeProgressCallback([this](float64 value) {
|
_playbackSlider->setChangeProgressCallback([this](float64 value) {
|
||||||
|
|
|
@ -140,8 +140,8 @@ void Player::trackPlayedTill(
|
||||||
if (guard && position != kTimeUnknown) {
|
if (guard && position != kTimeUnknown) {
|
||||||
state.position = position;
|
state.position = position;
|
||||||
const auto value = _options.loop
|
const auto value = _options.loop
|
||||||
? position
|
? (position % _totalDuration)
|
||||||
: (position % _totalDuration);
|
: position;
|
||||||
_updates.fire({ PlaybackUpdate<Track>{ value } });
|
_updates.fire({ PlaybackUpdate<Track>{ value } });
|
||||||
}
|
}
|
||||||
if (_pauseReading
|
if (_pauseReading
|
||||||
|
@ -159,9 +159,12 @@ void Player::trackSendReceivedTill(
|
||||||
Expects(state.duration != kTimeUnknown);
|
Expects(state.duration != kTimeUnknown);
|
||||||
Expects(state.receivedTill != kTimeUnknown);
|
Expects(state.receivedTill != kTimeUnknown);
|
||||||
|
|
||||||
|
const auto receivedTill = std::max(
|
||||||
|
state.receivedTill,
|
||||||
|
_previousReceivedTill);
|
||||||
const auto value = _options.loop
|
const auto value = _options.loop
|
||||||
? state.receivedTill
|
? (receivedTill % _totalDuration)
|
||||||
: (state.receivedTill % _totalDuration);
|
: receivedTill;
|
||||||
_updates.fire({ PreloadedUpdate<Track>{ value } });
|
_updates.fire({ PreloadedUpdate<Track>{ value } });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -376,9 +379,12 @@ void Player::play(const PlaybackOptions &options) {
|
||||||
// Looping video with audio is not supported for now.
|
// Looping video with audio is not supported for now.
|
||||||
Expects(!options.loop || (options.mode != Mode::Both));
|
Expects(!options.loop || (options.mode != Mode::Both));
|
||||||
|
|
||||||
|
const auto previous = getCurrentReceivedTill();
|
||||||
|
|
||||||
stop();
|
stop();
|
||||||
_lastFailureStage = Stage::Uninitialized;
|
_lastFailureStage = Stage::Uninitialized;
|
||||||
|
|
||||||
|
savePreviousReceivedTill(options, previous);
|
||||||
_options = options;
|
_options = options;
|
||||||
if (!Media::Audio::SupportsSpeedControl()) {
|
if (!Media::Audio::SupportsSpeedControl()) {
|
||||||
_options.speed = 1.;
|
_options.speed = 1.;
|
||||||
|
@ -387,6 +393,17 @@ void Player::play(const PlaybackOptions &options) {
|
||||||
_file->start(delegate(), _options.position);
|
_file->start(delegate(), _options.position);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Player::savePreviousReceivedTill(
|
||||||
|
const PlaybackOptions &options,
|
||||||
|
crl::time previousReceivedTill) {
|
||||||
|
// Save previous 'receivedTill' values if we seek inside the range.
|
||||||
|
_previousReceivedTill = ((options.position >= _options.position)
|
||||||
|
&& (options.mode == _options.mode)
|
||||||
|
&& (options.position < previousReceivedTill))
|
||||||
|
? previousReceivedTill
|
||||||
|
: kTimeUnknown;
|
||||||
|
}
|
||||||
|
|
||||||
void Player::pause() {
|
void Player::pause() {
|
||||||
Expects(active());
|
Expects(active());
|
||||||
|
|
||||||
|
@ -626,6 +643,7 @@ Media::Player::TrackState Player::prepareLegacyState() const {
|
||||||
} else if (_options.loop && _totalDuration > 0) {
|
} else if (_options.loop && _totalDuration > 0) {
|
||||||
result.position %= _totalDuration;
|
result.position %= _totalDuration;
|
||||||
}
|
}
|
||||||
|
result.receivedTill = getCurrentReceivedTill();
|
||||||
result.length = _totalDuration;
|
result.length = _totalDuration;
|
||||||
if (result.length == kTimeUnknown) {
|
if (result.length == kTimeUnknown) {
|
||||||
const auto document = _options.audioId.audio();
|
const auto document = _options.audioId.audio();
|
||||||
|
@ -644,6 +662,16 @@ Media::Player::TrackState Player::prepareLegacyState() const {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
crl::time Player::getCurrentReceivedTill() const {
|
||||||
|
const auto previous = std::max(_previousReceivedTill, crl::time(0));
|
||||||
|
const auto result = std::min(
|
||||||
|
std::max(_information.audio.state.receivedTill, previous),
|
||||||
|
std::max(_information.video.state.receivedTill, previous));
|
||||||
|
return (result >= 0 && _totalDuration > 1 && _options.loop)
|
||||||
|
? (result % _totalDuration)
|
||||||
|
: result;
|
||||||
|
}
|
||||||
|
|
||||||
rpl::lifetime &Player::lifetime() {
|
rpl::lifetime &Player::lifetime() {
|
||||||
return _lifetime;
|
return _lifetime;
|
||||||
}
|
}
|
||||||
|
|
|
@ -105,6 +105,10 @@ private:
|
||||||
[[nodiscard]] bool bothReceivedEnough(crl::time amount) const;
|
[[nodiscard]] bool bothReceivedEnough(crl::time amount) const;
|
||||||
[[nodiscard]] bool receivedTillEnd() const;
|
[[nodiscard]] bool receivedTillEnd() const;
|
||||||
void checkResumeFromWaitingForData();
|
void checkResumeFromWaitingForData();
|
||||||
|
[[nodiscard]] crl::time getCurrentReceivedTill() const;
|
||||||
|
void savePreviousReceivedTill(
|
||||||
|
const PlaybackOptions &options,
|
||||||
|
crl::time previousReceivedTill);
|
||||||
|
|
||||||
template <typename Track>
|
template <typename Track>
|
||||||
void trackReceivedTill(
|
void trackReceivedTill(
|
||||||
|
@ -153,14 +157,16 @@ private:
|
||||||
bool _audioFinished = false;
|
bool _audioFinished = false;
|
||||||
bool _videoFinished = false;
|
bool _videoFinished = false;
|
||||||
|
|
||||||
crl::time _totalDuration = 0;
|
|
||||||
crl::time _loopingShift = 0;
|
|
||||||
crl::time _startedTime = kTimeUnknown;
|
crl::time _startedTime = kTimeUnknown;
|
||||||
crl::time _pausedTime = kTimeUnknown;
|
crl::time _pausedTime = kTimeUnknown;
|
||||||
crl::time _nextFrameTime = kTimeUnknown;
|
crl::time _nextFrameTime = kTimeUnknown;
|
||||||
base::Timer _renderFrameTimer;
|
base::Timer _renderFrameTimer;
|
||||||
rpl::event_stream<Update, Error> _updates;
|
rpl::event_stream<Update, Error> _updates;
|
||||||
|
|
||||||
|
crl::time _totalDuration = 0;
|
||||||
|
crl::time _loopingShift = 0;
|
||||||
|
crl::time _previousReceivedTill = kTimeUnknown;
|
||||||
|
|
||||||
rpl::lifetime _lifetime;
|
rpl::lifetime _lifetime;
|
||||||
rpl::lifetime _sessionLifetime;
|
rpl::lifetime _sessionLifetime;
|
||||||
|
|
||||||
|
|
|
@ -2011,7 +2011,7 @@ void OverlayWidget::handleStreamingUpdate(Streaming::Update &&update) {
|
||||||
streamingReady(std::move(update));
|
streamingReady(std::move(update));
|
||||||
}, [&](const PreloadedVideo &update) {
|
}, [&](const PreloadedVideo &update) {
|
||||||
_streamed->info.video.state.receivedTill = update.till;
|
_streamed->info.video.state.receivedTill = update.till;
|
||||||
//updatePlaybackState();
|
updatePlaybackState();
|
||||||
}, [&](const UpdateVideo &update) {
|
}, [&](const UpdateVideo &update) {
|
||||||
_streamed->info.video.state.position = update.position;
|
_streamed->info.video.state.position = update.position;
|
||||||
this->update(contentRect());
|
this->update(contentRect());
|
||||||
|
@ -2019,7 +2019,7 @@ void OverlayWidget::handleStreamingUpdate(Streaming::Update &&update) {
|
||||||
updatePlaybackState();
|
updatePlaybackState();
|
||||||
}, [&](const PreloadedAudio &update) {
|
}, [&](const PreloadedAudio &update) {
|
||||||
_streamed->info.audio.state.receivedTill = update.till;
|
_streamed->info.audio.state.receivedTill = update.till;
|
||||||
//updatePlaybackState();
|
updatePlaybackState();
|
||||||
}, [&](const UpdateAudio &update) {
|
}, [&](const UpdateAudio &update) {
|
||||||
_streamed->info.audio.state.position = update.position;
|
_streamed->info.audio.state.position = update.position;
|
||||||
updatePlaybackState();
|
updatePlaybackState();
|
||||||
|
@ -2240,7 +2240,7 @@ void OverlayWidget::playbackControlsVolumeChanged(float64 volume) {
|
||||||
void OverlayWidget::playbackToggleFullScreen() {
|
void OverlayWidget::playbackToggleFullScreen() {
|
||||||
Expects(_streamed != nullptr);
|
Expects(_streamed != nullptr);
|
||||||
|
|
||||||
if (!videoShown()) {
|
if (!videoShown() || (videoIsGifv() && !_fullScreenVideo)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_fullScreenVideo = !_fullScreenVideo;
|
_fullScreenVideo = !_fullScreenVideo;
|
||||||
|
@ -2279,6 +2279,9 @@ void OverlayWidget::playbackResumeOnCall() {
|
||||||
void OverlayWidget::updatePlaybackState() {
|
void OverlayWidget::updatePlaybackState() {
|
||||||
Expects(_streamed != nullptr);
|
Expects(_streamed != nullptr);
|
||||||
|
|
||||||
|
if (videoIsGifv()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
auto state = _streamed->player.prepareLegacyState();
|
auto state = _streamed->player.prepareLegacyState();
|
||||||
if (state.length == kTimeUnknown) {
|
if (state.length == kTimeUnknown) {
|
||||||
const auto duration = _doc->song()
|
const auto duration = _doc->song()
|
||||||
|
|
|
@ -58,11 +58,10 @@ PlaybackControls::PlaybackControls(QWidget *parent, not_null<Delegate *> delegat
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
_playbackProgress->setInLoadingStateChangedCallback([=](bool loading) {
|
_playbackProgress->setValueChangedCallback([=](
|
||||||
_playbackSlider->setDisabled(loading);
|
float64 value,
|
||||||
});
|
float64 receivedTill) {
|
||||||
_playbackProgress->setValueChangedCallback([=](float64 value) {
|
_playbackSlider->setValue(value, receivedTill);
|
||||||
_playbackSlider->setValue(value);
|
|
||||||
});
|
});
|
||||||
_playbackSlider->setChangeProgressCallback([=](float64 value) {
|
_playbackSlider->setChangeProgressCallback([=](float64 value) {
|
||||||
_playbackProgress->setValue(value, false);
|
_playbackProgress->setValue(value, false);
|
||||||
|
|
|
@ -78,6 +78,7 @@ private:
|
||||||
object_ptr<Ui::IconButton> _playPauseResume;
|
object_ptr<Ui::IconButton> _playPauseResume;
|
||||||
object_ptr<Ui::MediaSlider> _playbackSlider;
|
object_ptr<Ui::MediaSlider> _playbackSlider;
|
||||||
std::unique_ptr<PlaybackProgress> _playbackProgress;
|
std::unique_ptr<PlaybackProgress> _playbackProgress;
|
||||||
|
std::unique_ptr<PlaybackProgress> _receivedTillProgress;
|
||||||
object_ptr<Ui::MediaSlider> _volumeController;
|
object_ptr<Ui::MediaSlider> _volumeController;
|
||||||
object_ptr<Ui::IconButton> _fullScreenToggle;
|
object_ptr<Ui::IconButton> _fullScreenToggle;
|
||||||
object_ptr<Ui::LabelSimple> _playedAlready;
|
object_ptr<Ui::LabelSimple> _playedAlready;
|
||||||
|
|
|
@ -18,13 +18,24 @@ constexpr auto kPlaybackAnimationDurationMs = crl::time(200);
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
PlaybackProgress::PlaybackProgress() : _a_value(animation(this, &PlaybackProgress::step_value)) {
|
PlaybackProgress::PlaybackProgress()
|
||||||
|
: _a_value(animation(this, &PlaybackProgress::step_value))
|
||||||
|
, _a_receivedTill(animation(this, &PlaybackProgress::step_receivedTill)) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void PlaybackProgress::updateState(const Player::TrackState &state) {
|
void PlaybackProgress::updateState(const Player::TrackState &state) {
|
||||||
qint64 position = 0, length = state.length;
|
_playing = !Player::IsStopped(state.state);
|
||||||
|
const auto length = state.length;
|
||||||
|
const auto position = Player::IsStoppedAtEnd(state.state)
|
||||||
|
? state.length
|
||||||
|
: Player::IsStoppedOrStopping(state.state)
|
||||||
|
? 0
|
||||||
|
: state.position;
|
||||||
|
const auto receivedTill = (length && state.receivedTill > position)
|
||||||
|
? state.receivedTill
|
||||||
|
: -1;
|
||||||
|
|
||||||
auto wasInLoadingState = _inLoadingState;
|
const auto wasInLoadingState = _inLoadingState;
|
||||||
if (wasInLoadingState) {
|
if (wasInLoadingState) {
|
||||||
_inLoadingState = false;
|
_inLoadingState = false;
|
||||||
if (_inLoadingStateChanged) {
|
if (_inLoadingStateChanged) {
|
||||||
|
@ -32,23 +43,16 @@ void PlaybackProgress::updateState(const Player::TrackState &state) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_playing = !Player::IsStopped(state.state);
|
const auto progress = (position > length)
|
||||||
if (Player::IsStoppedAtEnd(state.state)) {
|
? 1.
|
||||||
position = state.length;
|
: length
|
||||||
} else if (!Player::IsStoppedOrStopping(state.state)) {
|
? snap(float64(position) / length, 0., 1.)
|
||||||
position = state.position;
|
: 0.;
|
||||||
} else {
|
const auto receivedTillProgress = (receivedTill > position)
|
||||||
position = 0;
|
? snap(float64(receivedTill) / length, 0., 1.)
|
||||||
}
|
: -1.;
|
||||||
|
const auto animatedPosition = position + (state.frequency * kPlaybackAnimationDurationMs / 1000);
|
||||||
auto progress = 0.;
|
const auto animatedProgress = length ? qMax(float64(animatedPosition) / length, 0.) : 0.;
|
||||||
if (position > length) {
|
|
||||||
progress = 1.;
|
|
||||||
} else if (length) {
|
|
||||||
progress = snap(float64(position) / length, 0., 1.);
|
|
||||||
}
|
|
||||||
auto animatedPosition = position + (state.frequency * kPlaybackAnimationDurationMs / 1000);
|
|
||||||
auto animatedProgress = length ? qMax(float64(animatedPosition) / length, 0.) : 0.;
|
|
||||||
if (length != _length || position != _position || wasInLoadingState) {
|
if (length != _length || position != _position || wasInLoadingState) {
|
||||||
if (auto animated = (length && _length && animatedProgress > value())) {
|
if (auto animated = (length && _length && animatedProgress > value())) {
|
||||||
setValue(animatedProgress, animated);
|
setValue(animatedProgress, animated);
|
||||||
|
@ -58,6 +62,10 @@ void PlaybackProgress::updateState(const Player::TrackState &state) {
|
||||||
_position = position;
|
_position = position;
|
||||||
_length = length;
|
_length = length;
|
||||||
}
|
}
|
||||||
|
if (receivedTill != _receivedTill) {
|
||||||
|
setReceivedTill(receivedTillProgress);
|
||||||
|
_receivedTill = receivedTill;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void PlaybackProgress::updateLoadingState(float64 progress) {
|
void PlaybackProgress::updateLoadingState(float64 progress) {
|
||||||
|
@ -88,9 +96,22 @@ void PlaybackProgress::setValue(float64 value, bool animated) {
|
||||||
a_value = anim::value(value, value);
|
a_value = anim::value(value, value);
|
||||||
_a_value.stop();
|
_a_value.stop();
|
||||||
}
|
}
|
||||||
if (_valueChanged) {
|
emitUpdatedValue();
|
||||||
_valueChanged(a_value.current());
|
}
|
||||||
|
|
||||||
|
void PlaybackProgress::setReceivedTill(float64 value) {
|
||||||
|
const auto current = a_receivedTill.current();
|
||||||
|
if (value > current && current > 0.) {
|
||||||
|
a_receivedTill.start(value);
|
||||||
|
_a_receivedTill.start();
|
||||||
|
} else if (value > a_value.current()) {
|
||||||
|
a_receivedTill = anim::value(a_value.current(), value);
|
||||||
|
_a_receivedTill.start();
|
||||||
|
} else {
|
||||||
|
a_receivedTill = anim::value(-1., -1.);
|
||||||
|
_a_receivedTill.stop();
|
||||||
}
|
}
|
||||||
|
emitUpdatedValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
void PlaybackProgress::step_value(float64 ms, bool timer) {
|
void PlaybackProgress::step_value(float64 ms, bool timer) {
|
||||||
|
@ -101,8 +122,29 @@ void PlaybackProgress::step_value(float64 ms, bool timer) {
|
||||||
} else {
|
} else {
|
||||||
a_value.update(dt, anim::linear);
|
a_value.update(dt, anim::linear);
|
||||||
}
|
}
|
||||||
if (timer && _valueChanged) {
|
if (timer) {
|
||||||
_valueChanged(a_value.current());
|
emitUpdatedValue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PlaybackProgress::step_receivedTill(float64 ms, bool timer) {
|
||||||
|
auto dt = anim::Disabled() ? 1. : (ms / kPlaybackAnimationDurationMs);
|
||||||
|
if (dt >= 1.) {
|
||||||
|
_a_receivedTill.stop();
|
||||||
|
a_receivedTill.finish();
|
||||||
|
} else {
|
||||||
|
a_receivedTill.update(dt, anim::linear);
|
||||||
|
}
|
||||||
|
if (timer) {
|
||||||
|
emitUpdatedValue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PlaybackProgress::emitUpdatedValue() {
|
||||||
|
if (_valueChanged) {
|
||||||
|
const auto value = a_value.current();
|
||||||
|
const auto receivedTill = a_receivedTill.current();
|
||||||
|
_valueChanged(value, std::max(value, receivedTill));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,7 @@ class PlaybackProgress {
|
||||||
public:
|
public:
|
||||||
PlaybackProgress();
|
PlaybackProgress();
|
||||||
|
|
||||||
void setValueChangedCallback(Fn<void(float64)> callback) {
|
void setValueChangedCallback(Fn<void(float64,float64)> callback) {
|
||||||
_valueChanged = std::move(callback);
|
_valueChanged = std::move(callback);
|
||||||
}
|
}
|
||||||
void setInLoadingStateChangedCallback(Fn<void(bool)> callback) {
|
void setInLoadingStateChangedCallback(Fn<void(bool)> callback) {
|
||||||
|
@ -35,18 +35,23 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void step_value(float64 ms, bool timer);
|
void step_value(float64 ms, bool timer);
|
||||||
|
void step_receivedTill(float64 ms, bool timer);
|
||||||
|
void setReceivedTill(float64 value);
|
||||||
|
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 BasicAnimation, not an Animation.
|
// so it should be a BasicAnimation, not an Animation, because
|
||||||
anim::value a_value;
|
// Animation pauses mtproto responses/updates handling while playing.
|
||||||
BasicAnimation _a_value;
|
anim::value a_value, a_receivedTill;
|
||||||
Fn<void(float64)> _valueChanged;
|
BasicAnimation _a_value, _a_receivedTill;
|
||||||
|
Fn<void(float64,float64)> _valueChanged;
|
||||||
|
|
||||||
bool _inLoadingState = false;
|
bool _inLoadingState = false;
|
||||||
Fn<void(bool)> _inLoadingStateChanged;
|
Fn<void(bool)> _inLoadingStateChanged;
|
||||||
|
|
||||||
int64 _position = 0;
|
int64 _position = 0;
|
||||||
int64 _length = 0;
|
int64 _length = 0;
|
||||||
|
int64 _receivedTill = -1;
|
||||||
|
|
||||||
bool _playing = false;
|
bool _playing = false;
|
||||||
|
|
||||||
|
|
|
@ -16,9 +16,10 @@ mediaviewPlayback: MediaSlider {
|
||||||
activeFg: mediaviewPlaybackActive;
|
activeFg: mediaviewPlaybackActive;
|
||||||
inactiveFg: mediaviewPlaybackInactive;
|
inactiveFg: mediaviewPlaybackInactive;
|
||||||
activeFgOver: mediaviewPlaybackActiveOver;
|
activeFgOver: mediaviewPlaybackActiveOver;
|
||||||
inactiveFgOver: mediaviewPlaybackInactiveOver;
|
inactiveFgOver: mediaviewPlaybackInactive;
|
||||||
activeFgDisabled: mediaviewPlaybackActive;
|
activeFgDisabled: mediaviewPlaybackActive;
|
||||||
inactiveFgDisabled: mediaviewPlaybackInactive;
|
inactiveFgDisabled: mediaviewPlaybackInactive;
|
||||||
|
receivedTillFg: mediaviewPlaybackInactiveOver;
|
||||||
seekSize: size(11px, 11px);
|
seekSize: size(11px, 11px);
|
||||||
duration: mediaviewOverDuration;
|
duration: mediaviewOverDuration;
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,9 +42,26 @@ void ContinuousSlider::setMoveByWheel(bool move) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QRect ContinuousSlider::getSeekRect() const {
|
||||||
|
const auto decrease = getSeekDecreaseSize();
|
||||||
|
return isHorizontal()
|
||||||
|
? QRect(decrease.width() / 2, 0, width() - decrease.width(), height())
|
||||||
|
: QRect(0, decrease.height() / 2, width(), height() - decrease.width());
|
||||||
|
}
|
||||||
|
|
||||||
void ContinuousSlider::setValue(float64 value) {
|
void ContinuousSlider::setValue(float64 value) {
|
||||||
_value = value;
|
setValue(value, value);
|
||||||
update();
|
}
|
||||||
|
|
||||||
|
void ContinuousSlider::setValue(float64 value, float64 receivedTill) {
|
||||||
|
if (_value != value || _receivedTill != receivedTill) {
|
||||||
|
LOG(("UPDATED"));
|
||||||
|
_value = value;
|
||||||
|
_receivedTill = receivedTill;
|
||||||
|
update();
|
||||||
|
} else {
|
||||||
|
LOG(("SKIPPED"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ContinuousSlider::setFadeOpacity(float64 opacity) {
|
void ContinuousSlider::setFadeOpacity(float64 opacity) {
|
||||||
|
@ -141,8 +158,8 @@ FilledSlider::FilledSlider(QWidget *parent, const style::FilledSlider &st) : Con
|
||||||
, _st(st) {
|
, _st(st) {
|
||||||
}
|
}
|
||||||
|
|
||||||
QRect FilledSlider::getSeekRect() const {
|
QSize FilledSlider::getSeekDecreaseSize() const {
|
||||||
return QRect(0, 0, width(), height());
|
return QSize(0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
float64 FilledSlider::getOverDuration() const {
|
float64 FilledSlider::getOverDuration() const {
|
||||||
|
@ -155,16 +172,18 @@ void FilledSlider::paintEvent(QPaintEvent *e) {
|
||||||
|
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
|
|
||||||
auto masterOpacity = fadeOpacity();
|
const auto masterOpacity = fadeOpacity();
|
||||||
auto ms = crl::now();
|
const auto ms = crl::now();
|
||||||
auto disabled = isDisabled();
|
const auto disabled = isDisabled();
|
||||||
auto over = getCurrentOverFactor(ms);
|
const auto over = getCurrentOverFactor(ms);
|
||||||
auto lineWidth = _st.lineWidth + ((_st.fullWidth - _st.lineWidth) * over);
|
const auto lineWidth = _st.lineWidth + ((_st.fullWidth - _st.lineWidth) * over);
|
||||||
auto lineWidthRounded = qFloor(lineWidth);
|
const auto lineWidthRounded = qFloor(lineWidth);
|
||||||
auto lineWidthPartial = lineWidth - lineWidthRounded;
|
const auto lineWidthPartial = lineWidth - lineWidthRounded;
|
||||||
auto seekRect = getSeekRect();
|
const auto seekRect = getSeekRect();
|
||||||
auto value = getCurrentValue();
|
const auto value = getCurrentValue();
|
||||||
auto from = seekRect.x(), mid = qRound(from + value * seekRect.width()), end = from + seekRect.width();
|
const auto from = seekRect.x();
|
||||||
|
const auto mid = qRound(from + value * seekRect.width());
|
||||||
|
const auto end = from + seekRect.width();
|
||||||
if (mid > from) {
|
if (mid > from) {
|
||||||
p.setOpacity(masterOpacity);
|
p.setOpacity(masterOpacity);
|
||||||
p.fillRect(from, height() - lineWidthRounded, (mid - from), lineWidthRounded, disabled ? _st.disabledFg : _st.activeFg);
|
p.fillRect(from, height() - lineWidthRounded, (mid - from), lineWidthRounded, disabled ? _st.disabledFg : _st.activeFg);
|
||||||
|
@ -187,10 +206,8 @@ MediaSlider::MediaSlider(QWidget *parent, const style::MediaSlider &st) : Contin
|
||||||
, _st(st) {
|
, _st(st) {
|
||||||
}
|
}
|
||||||
|
|
||||||
QRect MediaSlider::getSeekRect() const {
|
QSize MediaSlider::getSeekDecreaseSize() const {
|
||||||
return isHorizontal()
|
return _alwaysDisplayMarker ? _st.seekSize : QSize();
|
||||||
? QRect(_st.seekSize.width() / 2, 0, width() - _st.seekSize.width(), height())
|
|
||||||
: QRect(0, _st.seekSize.height() / 2, width(), height() - _st.seekSize.width());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float64 MediaSlider::getOverDuration() const {
|
float64 MediaSlider::getOverDuration() const {
|
||||||
|
@ -211,57 +228,81 @@ void MediaSlider::paintEvent(QPaintEvent *e) {
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
p.setOpacity(fadeOpacity());
|
p.setOpacity(fadeOpacity());
|
||||||
|
|
||||||
auto horizontal = isHorizontal();
|
const auto horizontal = isHorizontal();
|
||||||
auto ms = crl::now();
|
const auto ms = crl::now();
|
||||||
auto radius = _st.width / 2;
|
const auto radius = _st.width / 2;
|
||||||
auto disabled = isDisabled();
|
const auto disabled = isDisabled();
|
||||||
auto over = getCurrentOverFactor(ms);
|
const auto over = getCurrentOverFactor(ms);
|
||||||
auto seekRect = getSeekRect();
|
const auto seekRect = getSeekRect();
|
||||||
auto value = getCurrentValue();
|
|
||||||
|
|
||||||
// invert colors and value for vertical
|
// invert colors and value for vertical
|
||||||
if (!horizontal) value = 1. - value;
|
const auto value = horizontal
|
||||||
|
? getCurrentValue()
|
||||||
|
: (1. - getCurrentValue());
|
||||||
|
|
||||||
auto markerFrom = (horizontal ? seekRect.x() : seekRect.y());
|
// receivedTill is not supported for vertical
|
||||||
auto markerLength = (horizontal ? seekRect.width() : seekRect.height());
|
const auto receivedTill = horizontal
|
||||||
auto from = _alwaysDisplayMarker ? 0 : markerFrom;
|
? getCurrentReceivedTill()
|
||||||
auto length = _alwaysDisplayMarker ? (horizontal ? width() : height()) : markerLength;
|
: value;
|
||||||
auto mid = qRound(from + value * length);
|
|
||||||
auto end = from + length;
|
const auto markerFrom = (horizontal ? seekRect.x() : seekRect.y());
|
||||||
auto activeFg = disabled ? _st.activeFgDisabled : anim::brush(_st.activeFg, _st.activeFgOver, over);
|
const auto markerLength = (horizontal ? seekRect.width() : seekRect.height());
|
||||||
auto inactiveFg = disabled ? _st.inactiveFgDisabled : anim::brush(_st.inactiveFg, _st.inactiveFgOver, over);
|
const auto from = 0;
|
||||||
|
const auto length = (horizontal ? width() : height());
|
||||||
|
const auto mid = qRound(from + value * length);
|
||||||
|
const auto till = std::max(mid, qRound(from + receivedTill * length));
|
||||||
|
const auto end = from + length;
|
||||||
|
const auto activeFg = disabled ? _st.activeFgDisabled : anim::brush(_st.activeFg, _st.activeFgOver, over);
|
||||||
|
const auto receivedTillFg = _st.receivedTillFg;
|
||||||
|
const auto inactiveFg = disabled ? _st.inactiveFgDisabled : anim::brush(_st.inactiveFg, _st.inactiveFgOver, over);
|
||||||
if (mid > from) {
|
if (mid > from) {
|
||||||
auto fromClipRect = horizontal ? QRect(0, 0, mid, height()) : QRect(0, 0, width(), mid);
|
const auto fromClipRect = horizontal ? QRect(0, 0, mid, height()) : QRect(0, 0, width(), mid);
|
||||||
const auto till = std::min(mid + radius, end);
|
const auto till = std::min(mid + radius, end);
|
||||||
auto fromRect = horizontal
|
const auto fromRect = horizontal
|
||||||
? QRect(from, (height() - _st.width) / 2, till - from, _st.width)
|
? QRect(from, (height() - _st.width) / 2, till - from, _st.width)
|
||||||
: QRect((width() - _st.width) / 2, from, _st.width, till - from);
|
: QRect((width() - _st.width) / 2, from, _st.width, till - from);
|
||||||
p.setClipRect(fromClipRect);
|
p.setClipRect(fromClipRect);
|
||||||
p.setBrush(horizontal ? activeFg : inactiveFg);
|
p.setBrush(horizontal ? activeFg : inactiveFg);
|
||||||
p.drawRoundedRect(fromRect, radius, radius);
|
p.drawRoundedRect(fromRect, radius, radius);
|
||||||
}
|
}
|
||||||
if (end > mid) {
|
if (till > mid) {
|
||||||
auto endClipRect = horizontal ? QRect(mid, 0, width() - mid, height()) : QRect(0, mid, width(), height() - mid);
|
Assert(horizontal);
|
||||||
const auto begin = std::max(mid - radius, from);
|
auto clipRect = QRect(mid, 0, till - mid, height());
|
||||||
auto endRect = horizontal
|
const auto left = std::max(mid - radius, from);
|
||||||
|
const auto right = std::min(till + radius, end);
|
||||||
|
const auto rect = QRect(left, (height() - _st.width) / 2, right - left, _st.width);
|
||||||
|
p.setClipRect(clipRect);
|
||||||
|
p.setBrush(receivedTillFg);
|
||||||
|
p.drawRoundedRect(rect, radius, radius);
|
||||||
|
}
|
||||||
|
if (end > till) {
|
||||||
|
const auto endClipRect = horizontal ? QRect(till, 0, width() - till, height()) : QRect(0, till, width(), height() - till);
|
||||||
|
const auto begin = std::max(till - radius, from);
|
||||||
|
const auto endRect = horizontal
|
||||||
? QRect(begin, (height() - _st.width) / 2, end - begin, _st.width)
|
? QRect(begin, (height() - _st.width) / 2, end - begin, _st.width)
|
||||||
: QRect((width() - _st.width) / 2, begin, _st.width, end - begin);
|
: QRect((width() - _st.width) / 2, begin, _st.width, end - begin);
|
||||||
p.setClipRect(endClipRect);
|
p.setClipRect(endClipRect);
|
||||||
p.setBrush(horizontal ? inactiveFg : activeFg);
|
p.setBrush(horizontal ? inactiveFg : activeFg);
|
||||||
p.drawRoundedRect(endRect, radius, radius);
|
p.drawRoundedRect(endRect, radius, radius);
|
||||||
}
|
}
|
||||||
auto markerSizeRatio = disabled ? 0. : (_alwaysDisplayMarker ? 1. : over);
|
const auto markerSizeRatio = disabled ? 0. : (_alwaysDisplayMarker ? 1. : over);
|
||||||
if (markerSizeRatio > 0) {
|
if (markerSizeRatio > 0) {
|
||||||
auto position = qRound(markerFrom + value * markerLength) - (horizontal ? seekRect.x() : seekRect.y());
|
const auto position = qRound(markerFrom + value * markerLength) - (horizontal ? (_st.seekSize.width() / 2) : (_st.seekSize.height() / 2));
|
||||||
auto seekButton = horizontal
|
const auto seekButton = horizontal
|
||||||
? QRect(position, (height() - _st.seekSize.height()) / 2, _st.seekSize.width(), _st.seekSize.height())
|
? QRect(position, (height() - _st.seekSize.height()) / 2, _st.seekSize.width(), _st.seekSize.height())
|
||||||
: QRect((width() - _st.seekSize.width()) / 2, position, _st.seekSize.width(), _st.seekSize.height());
|
: QRect((width() - _st.seekSize.width()) / 2, position, _st.seekSize.width(), _st.seekSize.height());
|
||||||
auto size = horizontal ? _st.seekSize.width() : _st.seekSize.height();
|
const auto size = horizontal ? _st.seekSize.width() : _st.seekSize.height();
|
||||||
auto remove = static_cast<int>(((1. - markerSizeRatio) * size) / 2.);
|
const auto remove = static_cast<int>(((1. - markerSizeRatio) * size) / 2.);
|
||||||
if (remove * 2 < size) {
|
if (remove * 2 < size) {
|
||||||
p.setClipRect(rect());
|
p.setClipRect(rect());
|
||||||
p.setBrush(activeFg);
|
p.setBrush(activeFg);
|
||||||
p.drawEllipse(seekButton.marginsRemoved(QMargins(remove, remove, remove, remove)));
|
const auto xshift = horizontal
|
||||||
|
? std::max(seekButton.x() + seekButton.width() - remove - width(), 0) + std::min(seekButton.x() + remove, 0)
|
||||||
|
: 0;
|
||||||
|
const auto yshift = horizontal
|
||||||
|
? 0
|
||||||
|
: std::max(seekButton.y() + seekButton.height() - remove - height(), 0) + std::min(seekButton.y() + remove, 0);
|
||||||
|
p.drawEllipse(seekButton.marginsRemoved(QMargins(remove, remove, remove, remove)).translated(-xshift, -yshift));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,6 +31,7 @@ public:
|
||||||
|
|
||||||
float64 value() const;
|
float64 value() const;
|
||||||
void setValue(float64 value);
|
void setValue(float64 value);
|
||||||
|
void setValue(float64 value, float64 receivedTill);
|
||||||
void setFadeOpacity(float64 opacity);
|
void setFadeOpacity(float64 opacity);
|
||||||
void setDisabled(bool disabled);
|
void setDisabled(bool disabled);
|
||||||
bool isDisabled() const {
|
bool isDisabled() const {
|
||||||
|
@ -63,9 +64,12 @@ protected:
|
||||||
float64 fadeOpacity() const {
|
float64 fadeOpacity() const {
|
||||||
return _fadeOpacity;
|
return _fadeOpacity;
|
||||||
}
|
}
|
||||||
float64 getCurrentValue() {
|
float64 getCurrentValue() const {
|
||||||
return _mouseDown ? _downValue : _value;
|
return _mouseDown ? _downValue : _value;
|
||||||
}
|
}
|
||||||
|
float64 getCurrentReceivedTill() const {
|
||||||
|
return _receivedTill;
|
||||||
|
}
|
||||||
float64 getCurrentOverFactor(crl::time ms) {
|
float64 getCurrentOverFactor(crl::time ms) {
|
||||||
return _disabled ? 0. : _a_over.current(ms, _over ? 1. : 0.);
|
return _disabled ? 0. : _a_over.current(ms, _over ? 1. : 0.);
|
||||||
}
|
}
|
||||||
|
@ -75,9 +79,10 @@ protected:
|
||||||
bool isHorizontal() const {
|
bool isHorizontal() const {
|
||||||
return (_direction == Direction::Horizontal);
|
return (_direction == Direction::Horizontal);
|
||||||
}
|
}
|
||||||
|
QRect getSeekRect() const;
|
||||||
|
virtual QSize getSeekDecreaseSize() const = 0;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
virtual QRect getSeekRect() const = 0;
|
|
||||||
virtual float64 getOverDuration() const = 0;
|
virtual float64 getOverDuration() const = 0;
|
||||||
|
|
||||||
bool moveByWheel() const {
|
bool moveByWheel() const {
|
||||||
|
@ -101,6 +106,7 @@ private:
|
||||||
Animation _a_over;
|
Animation _a_over;
|
||||||
|
|
||||||
float64 _value = 0.;
|
float64 _value = 0.;
|
||||||
|
float64 _receivedTill = 0.;
|
||||||
|
|
||||||
bool _mouseDown = false;
|
bool _mouseDown = false;
|
||||||
float64 _downValue = 0.;
|
float64 _downValue = 0.;
|
||||||
|
@ -117,7 +123,7 @@ protected:
|
||||||
void paintEvent(QPaintEvent *e) override;
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QRect getSeekRect() const override;
|
QSize getSeekDecreaseSize() const override;
|
||||||
float64 getOverDuration() const override;
|
float64 getOverDuration() const override;
|
||||||
|
|
||||||
const style::FilledSlider &_st;
|
const style::FilledSlider &_st;
|
||||||
|
@ -176,7 +182,7 @@ protected:
|
||||||
void paintEvent(QPaintEvent *e) override;
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QRect getSeekRect() const override;
|
QSize getSeekDecreaseSize() const override;
|
||||||
float64 getOverDuration() const override;
|
float64 getOverDuration() const override;
|
||||||
|
|
||||||
const style::MediaSlider &_st;
|
const style::MediaSlider &_st;
|
||||||
|
|
|
@ -291,6 +291,7 @@ MediaSlider {
|
||||||
inactiveFgOver: color;
|
inactiveFgOver: color;
|
||||||
activeFgDisabled: color;
|
activeFgDisabled: color;
|
||||||
inactiveFgDisabled: color;
|
inactiveFgDisabled: color;
|
||||||
|
receivedTillFg: color;
|
||||||
seekSize: size;
|
seekSize: size;
|
||||||
duration: int;
|
duration: int;
|
||||||
}
|
}
|
||||||
|
@ -906,6 +907,7 @@ defaultContinuousSlider: MediaSlider {
|
||||||
inactiveFgOver: mediaPlayerInactiveFg;
|
inactiveFgOver: mediaPlayerInactiveFg;
|
||||||
activeFgDisabled: mediaPlayerInactiveFg;
|
activeFgDisabled: mediaPlayerInactiveFg;
|
||||||
inactiveFgDisabled: windowBg;
|
inactiveFgDisabled: windowBg;
|
||||||
|
receivedTillFg: mediaPlayerInactiveFg;
|
||||||
seekSize: size(9px, 9px);
|
seekSize: size(9px, 9px);
|
||||||
duration: 150;
|
duration: 150;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue