Display radial playback progress in round videos.

This commit is contained in:
John Preston 2017-05-18 20:20:07 +03:00
parent 87ff770020
commit 5915f3f928
8 changed files with 75 additions and 16 deletions

View File

@ -395,6 +395,8 @@ historyFileThumbIconFgSelected: msgInBgSelected; // selected file with thumbnail
historyFileThumbRadialFg: historyFileThumbIconFg; // file with thumbnail (or photo / video) radial download animation line historyFileThumbRadialFg: historyFileThumbIconFg; // file with thumbnail (or photo / video) radial download animation line
historyFileThumbRadialFgSelected: historyFileThumbIconFgSelected; // selected file with thumbnail (or photo / video) radial download animation line historyFileThumbRadialFgSelected: historyFileThumbIconFgSelected; // selected file with thumbnail (or photo / video) radial download animation line
historyVideoMessageProgressFg: historyFileThumbIconFg; // radial playback progress in round video messages
msgWaveformInActive: windowBgActive; // inbox voice message active waveform lines (like played part of currently playing voice message) msgWaveformInActive: windowBgActive; // inbox voice message active waveform lines (like played part of currently playing voice message)
msgWaveformInActiveSelected: #51a3d3; // inbox selected voice message active waveform lines (like played part of currently playing voice message) msgWaveformInActiveSelected: #51a3d3; // inbox selected voice message active waveform lines (like played part of currently playing voice message)
msgWaveformInInactive: #d4dee6; // inbox voice message inactive waveform lines (like upcoming part of currently playing voice message) msgWaveformInInactive: #d4dee6; // inbox voice message inactive waveform lines (like upcoming part of currently playing voice message)

View File

@ -444,3 +444,4 @@ msgWaveformMax: 20px;
historyVideoMessageMute: icon {{ "volume_mute", historyFileThumbIconFg }}; historyVideoMessageMute: icon {{ "volume_mute", historyFileThumbIconFg }};
historyVideoMessageMuteSelected: icon {{ "volume_mute", historyFileThumbIconFgSelected }}; historyVideoMessageMuteSelected: icon {{ "volume_mute", historyFileThumbIconFgSelected }};
historyVideoMessageMuteSize: 25px; historyVideoMessageMuteSize: 25px;
historyVideoMessageProgressOpacity: 0.72;

View File

@ -27,6 +27,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "media/media_audio.h" #include "media/media_audio.h"
#include "media/media_clip_reader.h" #include "media/media_clip_reader.h"
#include "media/player/media_player_instance.h" #include "media/player/media_player_instance.h"
#include "media/view/media_clip_playback.h"
#include "boxes/confirm_box.h" #include "boxes/confirm_box.h"
#include "boxes/add_contact_box.h" #include "boxes/add_contact_box.h"
#include "core/click_handler_types.h" #include "core/click_handler_types.h"
@ -1959,9 +1960,35 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
} }
} }
p.drawPixmap(rthumb.topLeft(), _gif->current(_thumbw, _thumbh, usew, height, roundRadius, roundCorners, paused ? 0 : ms)); p.drawPixmap(rthumb.topLeft(), _gif->current(_thumbw, _thumbh, usew, height, roundRadius, roundCorners, paused ? 0 : ms));
if (displayMute) {
_roundPlayback.reset();
} else if (_roundPlayback) {
auto value = _roundPlayback->value();
if (value > 0.) {
auto pen = st::historyVideoMessageProgressFg->p;
auto was = p.pen();
pen.setWidth(st::radialLine);
pen.setCapStyle(Qt::RoundCap);
p.setPen(pen);
p.setOpacity(st::historyVideoMessageProgressOpacity);
auto from = QuarterArcLength;
auto len = -qRound(FullArcLength * value);
auto stepInside = st::radialLine / 2;
{
PainterHighQualityEnabler hq(p);
p.drawArc(rthumb.marginsRemoved(QMargins(stepInside, stepInside, stepInside, stepInside)), from, len);
}
p.setPen(was);
p.setOpacity(1.);
}
}
} else { } else {
p.drawPixmap(rthumb.topLeft(), _data->thumb->pixBlurredSingle(_thumbw, _thumbh, usew, height, roundRadius, roundCorners)); p.drawPixmap(rthumb.topLeft(), _data->thumb->pixBlurredSingle(_thumbw, _thumbh, usew, height, roundRadius, roundCorners));
} }
if (selected) { if (selected) {
App::complexOverlayRect(p, rthumb, roundRadius, roundCorners); App::complexOverlayRect(p, rthumb, roundRadius, roundCorners);
} }
@ -2296,6 +2323,8 @@ void HistoryGif::updateStatusText() const {
} else if (_data->loaded()) { } else if (_data->loaded()) {
statusSize = FileStatusSizeLoaded; statusSize = FileStatusSizeLoaded;
if (_gif && _gif->mode() == Media::Clip::Reader::Mode::Video) { if (_gif && _gif->mode() == Media::Clip::Reader::Mode::Video) {
statusSize = -1 - _data->duration();
auto state = Media::Player::mixer()->currentState(AudioMsgId::Type::Video); auto state = Media::Player::mixer()->currentState(AudioMsgId::Type::Video);
if (state.length) { if (state.length) {
auto position = int64(0); auto position = int64(0);
@ -2304,9 +2333,10 @@ void HistoryGif::updateStatusText() const {
} else if (state.state == Media::Player::State::StoppedAtEnd) { } else if (state.state == Media::Player::State::StoppedAtEnd) {
position = state.length; position = state.length;
} }
statusSize = -1 - ((state.length - position) / state.frequency); accumulate_max(statusSize, -1 - int((state.length - position) / state.frequency + 1));
} else { }
statusSize = -1 - _data->duration(); if (_roundPlayback) {
_roundPlayback->updateState(state);
} }
} }
} else { } else {
@ -2381,6 +2411,10 @@ bool HistoryGif::playInline(bool autoplay) {
_parent->clipCallback(notification); _parent->clipCallback(notification);
}, mode)); }, mode));
if (mode == Mode::Video) { if (mode == Mode::Video) {
_roundPlayback = std::make_unique<Media::Clip::Playback>();
_roundPlayback->setValueChangedCallback([this](float64 value) {
Ui::repaintHistoryItem(_parent);
});
if (App::main()) { if (App::main()) {
App::main()->mediaMarkRead(_data); App::main()->mediaMarkRead(_data);
} }

View File

@ -22,6 +22,12 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/effects/radial_animation.h" #include "ui/effects/radial_animation.h"
namespace Media {
namespace Clip {
class Playback;
} // namespace Clip
} // namespace Media
void historyInitMedia(); void historyInitMedia();
class HistoryFileMedia : public HistoryMedia { class HistoryFileMedia : public HistoryMedia {
@ -585,6 +591,7 @@ private:
int32 _thumbh = 1; int32 _thumbh = 1;
Text _caption; Text _caption;
mutable std::unique_ptr<Media::Clip::Playback> _roundPlayback;
Media::Clip::ReaderPointer _gif; Media::Clip::ReaderPointer _gif;
void setStatusSize(int32 newSize) const; void setStatusSize(int32 newSize) const;

View File

@ -312,9 +312,15 @@ void Mixer::Track::reattach(AudioMsgId::Type type) {
} }
alSourcei(stream.source, AL_SAMPLE_OFFSET, qMax(state.position - bufferedPosition, 0LL)); alSourcei(stream.source, AL_SAMPLE_OFFSET, qMax(state.position - bufferedPosition, 0LL));
if (IsActive(state.state)) { if (!IsStopped(state.state)) {
alSourcef(stream.source, AL_GAIN, ComputeVolume(type)); alSourcef(stream.source, AL_GAIN, ComputeVolume(type));
alSourcePlay(stream.source); alSourcePlay(stream.source);
if (IsPaused(state.state)) {
// We must always start the source if we want the AL_SAMPLE_OFFSET to be applied.
// Otherwise it won't be read by alGetSource and we'll get a corrupt position.
// So in case of a paused source we start it and then immediately pause it.
alSourcePause(stream.source);
}
} }
} }
@ -539,7 +545,6 @@ void Mixer::resetFadeStartPosition(AudioMsgId::Type type, int positionInBuffered
if (track->isStreamCreated()) { if (track->isStreamCreated()) {
ALint currentPosition = 0; ALint currentPosition = 0;
alGetSourcei(track->stream.source, AL_SAMPLE_OFFSET, &currentPosition); alGetSourcei(track->stream.source, AL_SAMPLE_OFFSET, &currentPosition);
if (Audio::PlaybackErrorHappened()) { if (Audio::PlaybackErrorHappened()) {
setStoppedState(track, State::StoppedAtError); setStoppedState(track, State::StoppedAtError);
onError(track->state.id); onError(track->state.id);

View File

@ -25,6 +25,11 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
namespace Media { namespace Media {
namespace Clip { namespace Clip {
namespace {
constexpr auto kPlaybackAnimationDurationMs = TimeMs(200);
} // namespace
Playback::Playback() : _a_value(animation(this, &Playback::step_value)) { Playback::Playback() : _a_value(animation(this, &Playback::step_value)) {
} }
@ -53,11 +58,16 @@ void Playback::updateState(const Player::TrackState &state) {
if (position > length) { if (position > length) {
progress = 1.; progress = 1.;
} else if (length) { } else if (length) {
progress = length ? snap(float64(position) / length, 0., 1.) : 0.; 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) {
auto animated = (length && _length && progress > value()); if (auto animated = (length && _length && animatedProgress > value())) {
setValue(progress, animated); setValue(animatedProgress, animated);
} else {
setValue(progress, animated);
}
_position = position; _position = position;
_length = length; _length = length;
} }
@ -74,9 +84,8 @@ void Playback::updateLoadingState(float64 progress) {
setValue(progress, animated); setValue(progress, animated);
} }
float64 Playback::value() const { float64 Playback::value() const {
return a_value.current(); return qMin(a_value.current(), 1.);
} }
void Playback::setValue(float64 value, bool animated) { void Playback::setValue(float64 value, bool animated) {
@ -93,12 +102,12 @@ void Playback::setValue(float64 value, bool animated) {
} }
void Playback::step_value(float64 ms, bool timer) { void Playback::step_value(float64 ms, bool timer) {
auto dt = ms / (2 * AudioVoiceMsgUpdateView); auto dt = ms / kPlaybackAnimationDurationMs;
if (dt >= 1) { if (dt >= 1.) {
_a_value.stop(); _a_value.stop();
a_value.finish(); a_value.finish();
} else { } else {
a_value.update(qMin(dt, 1.), anim::linear); a_value.update(dt, anim::linear);
} }
if (timer && _valueChanged) { if (timer && _valueChanged) {
_valueChanged(a_value.current()); _valueChanged(a_value.current());

View File

@ -40,12 +40,12 @@ public:
_inLoadingStateChanged = std::move(callback); _inLoadingStateChanged = std::move(callback);
} }
void setValue(float64 value, bool animated); void setValue(float64 value, bool animated);
float64 value() const;
void updateState(const Player::TrackState &state); void updateState(const Player::TrackState &state);
void updateLoadingState(float64 progress); void updateLoadingState(float64 progress);
private: private:
float64 value() const;
void step_value(float64 ms, bool timer); void step_value(float64 ms, bool timer);
// This can animate for a very long time (like in music playing), // This can animate for a very long time (like in music playing),

View File

@ -70,10 +70,11 @@ void RadialAnimation::step(TimeMs ms) {
} }
void RadialAnimation::draw(Painter &p, const QRect &inner, int32 thickness, style::color color) { void RadialAnimation::draw(Painter &p, const QRect &inner, int32 thickness, style::color color) {
float64 o = p.opacity(); auto o = p.opacity();
p.setOpacity(o * _opacity); p.setOpacity(o * _opacity);
QPen pen(color->p), was(p.pen()); auto pen = color->p;
auto was = p.pen();
pen.setWidth(thickness); pen.setWidth(thickness);
pen.setCapStyle(Qt::RoundCap); pen.setCapStyle(Qt::RoundCap);
p.setPen(pen); p.setPen(pen);