Video play progress displayed in MediaView (in case of audio stream).

This commit is contained in:
John Preston 2016-07-12 14:38:16 +03:00
parent d806d079a2
commit 01d448c1bd
17 changed files with 178 additions and 30 deletions

View File

@ -718,6 +718,14 @@ void AudioPlayer::stopAndClear() {
}
}
AudioPlaybackState AudioPlayer::currentVideoState(uint64 videoPlayId) {
QMutexLocker lock(&playerMutex);
auto current = dataForType(AudioMsgId::Type::Video);
if (!current || current->videoPlayId != videoPlayId) return AudioPlaybackState();
return current->playbackState;
}
AudioPlaybackState AudioPlayer::currentState(AudioMsgId *audio, AudioMsgId::Type type) {
QMutexLocker lock(&playerMutex);
auto current = dataForType(type);

View File

@ -71,6 +71,7 @@ public:
void feedFromVideo(VideoSoundPart &&part);
int64 getVideoCorrectedTime(uint64 playId, uint64 systemMs);
void videoSoundProgress(const AudioMsgId &audio);
AudioPlaybackState currentVideoState(uint64 videoPlayId);
void stopAndClear();
@ -215,12 +216,10 @@ class AudioPlayerFader : public QObject {
Q_OBJECT
public:
AudioPlayerFader(QThread *thread);
void resumeDevice();
signals:
void error(const AudioMsgId &audio);
void playPositionUpdated(const AudioMsgId &audio);
void audioStopped(const AudioMsgId &audio);
@ -228,8 +227,7 @@ signals:
void stopPauseDevice();
public slots:
public slots:
void onInit();
void onTimer();
void onPauseTimer();
@ -241,7 +239,6 @@ signals:
void onSongVolumeChanged();
private:
enum {
EmitError = 0x01,
EmitStopped = 0x02,

View File

@ -28,7 +28,8 @@ namespace Media {
namespace Clip {
namespace internal {
FFMpegReaderImplementation::FFMpegReaderImplementation(FileLocation *location, QByteArray *data) : ReaderImplementation(location, data) {
FFMpegReaderImplementation::FFMpegReaderImplementation(FileLocation *location, QByteArray *data, uint64 playId) : ReaderImplementation(location, data)
, _playId(playId) {
_frame = av_frame_alloc();
av_init_packet(&_packetNull);
_packetNull.data = nullptr;
@ -311,7 +312,6 @@ bool FFMpegReaderImplementation::start(Mode mode) {
} else {
soundData->length = (_fmtContext->streams[_audioStreamId]->duration * soundData->frequency * _fmtContext->streams[_audioStreamId]->time_base.num) / _fmtContext->streams[_audioStreamId]->time_base.den;
}
_playId = rand_value<uint64>();
audioPlayer()->playFromVideo(AudioMsgId(AudioMsgId::Type::Video), _playId, std_::move(soundData), 0);
}

View File

@ -35,7 +35,7 @@ namespace internal {
class FFMpegReaderImplementation : public ReaderImplementation {
public:
FFMpegReaderImplementation(FileLocation *location, QByteArray *data);
FFMpegReaderImplementation(FileLocation *location, QByteArray *data, uint64 playId);
bool readFramesTill(int64 ms) override;
uint64 framePresentationTime() const override;

View File

@ -87,7 +87,8 @@ QPixmap _prepareFrame(const FrameRequest &request, const QImage &original, bool
Reader::Reader(const FileLocation &location, const QByteArray &data, Callback &&callback, Mode mode)
: _callback(std_::move(callback))
, _mode(mode) {
, _mode(mode)
, _playId(rand_value<uint64>()) {
if (threads.size() < ClipThreadsCount) {
_threadIndex = threads.size();
threads.push_back(new QThread());
@ -289,6 +290,7 @@ class ReaderPrivate {
public:
ReaderPrivate(Reader *reader, const FileLocation &location, const QByteArray &data) : _interface(reader)
, _mode(reader->mode())
, _playId(reader->playId())
, _data(data)
, _location(_data.isEmpty() ? new FileLocation(location) : 0) {
if (_data.isEmpty() && !_location->accessEnable()) {
@ -364,7 +366,7 @@ public:
}
}
_implementation = std_::make_unique<internal::FFMpegReaderImplementation>(_location, &_data);
_implementation = std_::make_unique<internal::FFMpegReaderImplementation>(_location, &_data, _playId);
// _implementation = new QtGifReaderImplementation(_location, &_data);
auto implementationMode = [this]() {
@ -410,6 +412,7 @@ private:
Reader *_interface;
State _state = State::Reading;
Reader::Mode _mode;
uint64 _playId;
QByteArray _data;
FileLocation *_location;
@ -687,7 +690,8 @@ MTPDocumentAttribute readAttributes(const QString &fname, const QByteArray &data
FileLocation localloc(StorageFilePartial, fname);
QByteArray localdata(data);
auto reader = std_::make_unique<internal::FFMpegReaderImplementation>(&localloc, &localdata);
auto playId = 0ULL;
auto reader = std_::make_unique<internal::FFMpegReaderImplementation>(&localloc, &localdata, playId);
if (reader->start(internal::ReaderImplementation::Mode::OnlyGifv)) {
bool hasAlpha = false;
if (reader->readFramesTill(-1) && reader->renderFrame(cover, hasAlpha, QSize())) {

View File

@ -51,7 +51,6 @@ enum ReaderSteps {
class ReaderPrivate;
class Reader {
public:
using Callback = Function<void, Notification>;
enum class Mode {
Gif,
@ -68,6 +67,10 @@ public:
return _autoplay;
}
uint64 playId() const {
return _playId;
}
void start(int framew, int frameh, int outerw, int outerh, bool rounded);
QPixmap current(int framew, int frameh, int outerw, int outerh, uint64 ms);
QPixmap frameOriginal() const {
@ -114,6 +117,8 @@ private:
State _state = State::Reading;
uint64 _playId;
mutable int _width = 0;
mutable int _height = 0;

View File

@ -41,6 +41,8 @@ Controller::Controller(QWidget *parent) : TWidget(parent)
, _toPlayLeft(this, st::mediaviewPlayProgressLabel)
, _fadeAnimation(std_::make_unique<Ui::FadeAnimation>(this)) {
_fadeAnimation->show();
_fadeAnimation->setFinishedCallback(func(this, &Controller::fadeFinished));
_fadeAnimation->setUpdatedCallback(func(this, &Controller::fadeUpdated));
connect(_playPauseResume, SIGNAL(clicked()), this, SIGNAL(playPressed()));
connect(_fullScreenToggle, SIGNAL(clicked()), this, SIGNAL(toFullScreenPressed()));
connect(_playback, SIGNAL(seekProgress(int64)), this, SLOT(onSeekProgress(int64)));
@ -59,14 +61,38 @@ void Controller::onSeekFinished(int64 position) {
}
void Controller::showAnimated() {
_fadeAnimation->fadeIn(st::mvShowDuration);
startFading([this]() {
_fadeAnimation->fadeIn(st::mvShowDuration);
});
}
void Controller::hideAnimated() {
_fadeAnimation->fadeOut(st::mvHideDuration);
startFading([this]() {
_fadeAnimation->fadeOut(st::mvShowDuration);
});
}
template <typename Callback>
void Controller::startFading(Callback start) {
start();
_playback->show();
}
void Controller::fadeFinished() {
fadeUpdated(1.);
}
void Controller::fadeUpdated(float64 opacity) {
_playback->setFadeOpacity(opacity);
}
void Controller::updatePlayback(const AudioPlaybackState &playbackState) {
updatePlayPauseResumeState(playbackState);
_playback->updateState(playbackState);
updateTimeTexts(playbackState);
}
void Controller::updatePlayPauseResumeState(const AudioPlaybackState &playbackState) {
bool showPause = (playbackState.state == AudioPlayerPlaying || playbackState.state == AudioPlayerResuming);
if (showPause != _showPause) {
disconnect(_playPauseResume, SIGNAL(clicked()), this, _showPause ? SIGNAL(pausePressed()) : SIGNAL(playPressed()));
@ -75,8 +101,32 @@ void Controller::updatePlayback(const AudioPlaybackState &playbackState) {
_playPauseResume->setIcon(_showPause ? &st::mediaviewPauseIcon : nullptr);
}
}
_playback->updateState(playbackState);
void Controller::updateTimeTexts(const AudioPlaybackState &playbackState) {
qint64 position = 0, duration = playbackState.duration;
if (!(playbackState.state & AudioPlayerStoppedMask) && playbackState.state != AudioPlayerFinishing) {
position = playbackState.position;
} else if (playbackState.state == AudioPlayerStoppedAtEnd) {
position = playbackState.duration;
} else {
position = 0;
}
auto playFrequency = (playbackState.frequency ? playbackState.frequency : AudioVoiceMsgFrequency);
auto playAlready = position / playFrequency;
auto playLeft = (playbackState.duration / playFrequency) - playAlready;
auto timeAlready = formatDurationText(playAlready);
auto minus = QChar(8722);
auto timeLeft = minus + formatDurationText(playLeft);
auto alreadyChanged = false, leftChanged = false;
_playedAlready->setText(timeAlready, &alreadyChanged);
_toPlayLeft->setText(timeLeft, &leftChanged);
if (alreadyChanged || leftChanged) {
_fadeAnimation->refreshCache();
}
}
void Controller::setInFullScreen(bool inFullScreen) {
@ -88,18 +138,29 @@ void Controller::setInFullScreen(bool inFullScreen) {
connect(_fullScreenToggle, SIGNAL(clicked()), this, handler);
}
void Controller::grabStart() {
showChildren();
_playback->hide();
}
void Controller::grabFinish() {
hideChildren();
_playback->show();
}
void Controller::resizeEvent(QResizeEvent *e) {
int playTop = (height() - _playPauseResume->height()) / 2;
_playPauseResume->moveToLeft(playTop, playTop);
_playedAlready->moveToLeft(playTop + _playPauseResume->width() + playTop, 0);
int fullScreenTop = (height() - _fullScreenToggle->height()) / 2;
_fullScreenToggle->moveToRight(fullScreenTop, fullScreenTop);
_toPlayLeft->moveToRight(fullScreenTop + _fullScreenToggle->width() + fullScreenTop, 0);
_volumeController->moveToRight(fullScreenTop + _fullScreenToggle->width() + fullScreenTop, (height() - _volumeController->height()) / 2);
_playback->resize(width() - playTop - _playPauseResume->width() - playTop - fullScreenTop - _volumeController->width() - fullScreenTop - _fullScreenToggle->width() - fullScreenTop, _volumeController->height());
_playback->moveToLeft(playTop + _playPauseResume->width() + playTop, (height() - _playback->height()) / 2);
_playback->moveToLeft(playTop + _playPauseResume->width() + playTop, st::mediaviewPlaybackTop);
_playedAlready->moveToLeft(playTop + _playPauseResume->width() + playTop, st::mediaviewPlayProgressTop);
_toPlayLeft->moveToRight(width() - (playTop + _playPauseResume->width() + playTop) - _playback->width(), st::mediaviewPlayProgressTop);
}
void Controller::paintEvent(QPaintEvent *e) {

View File

@ -46,6 +46,9 @@ public:
void updatePlayback(const AudioPlaybackState &playbackState);
void setInFullScreen(bool inFullScreen);
void grabStart() override;
void grabFinish() override;
signals:
void playPressed();
void pausePressed();
@ -64,6 +67,14 @@ protected:
void paintEvent(QPaintEvent *e) override;
private:
template <typename Callback>
void startFading(Callback start);
void fadeFinished();
void fadeUpdated(float64 opacity);
void updatePlayPauseResumeState(const AudioPlaybackState &playbackState);
void updateTimeTexts(const AudioPlaybackState &playbackState);
bool _showPause = false;
int64 _seekPosition = -1;

View File

@ -60,6 +60,11 @@ void Playback::updateState(const AudioPlaybackState &playbackState) {
}
}
void Playback::setFadeOpacity(float64 opacity) {
_fadeOpacity = opacity;
update();
}
void Playback::step_progress(float64 ms, bool timer) {
float64 dt = ms / (2 * AudioVoiceMsgUpdateView);
if (_duration && dt >= 1) {
@ -75,6 +80,7 @@ void Playback::paintEvent(QPaintEvent *e) {
Painter p(this);
int radius = st::mediaviewPlaybackWidth / 2;
p.setOpacity(_fadeOpacity);
p.setPen(Qt::NoPen);
p.setRenderHint(QPainter::HighQualityAntialiasing);
@ -85,19 +91,19 @@ void Playback::paintEvent(QPaintEvent *e) {
int32 from = skip, mid = qRound(from + prg * length), end = from + length;
if (mid > from) {
p.setClipRect(0, 0, mid, height());
p.setOpacity(over * st::mediaviewActiveOpacity + (1. - over) * st::mediaviewInactiveOpacity);
p.setOpacity(_fadeOpacity * (over * st::mediaviewActiveOpacity + (1. - over) * st::mediaviewInactiveOpacity));
p.setBrush(st::mediaviewPlaybackActive);
p.drawRoundedRect(0, (height() - st::mediaviewPlaybackWidth) / 2, mid + radius, st::mediaviewPlaybackWidth, radius, radius);
}
if (end > mid) {
p.setClipRect(mid, 0, width() - mid, height());
p.setOpacity(1.);
p.setOpacity(_fadeOpacity);
p.setBrush(st::mediaviewPlaybackInactive);
p.drawRoundedRect(mid - radius, (height() - st::mediaviewPlaybackWidth) / 2, width() - (mid - radius), st::mediaviewPlaybackWidth, radius, radius);
}
int x = mid - skip;
p.setClipRect(rect());
p.setOpacity(over * st::mediaviewActiveOpacity + (1. - over) * st::mediaviewInactiveOpacity);
p.setOpacity(_fadeOpacity * (over * st::mediaviewActiveOpacity + (1. - over) * st::mediaviewInactiveOpacity));
p.setBrush(st::mediaviewPlaybackActive);
p.drawRoundedRect(x, (height() - st::mediaviewSeekSize.height()) / 2, st::mediaviewSeekSize.width(), st::mediaviewSeekSize.height(), st::mediaviewSeekSize.width() / 2, st::mediaviewSeekSize.width() / 2);
}

View File

@ -32,6 +32,7 @@ public:
Playback(QWidget *parent);
void updateState(const AudioPlaybackState &playbackState);
void setFadeOpacity(float64 opacity);
signals:
void seekProgress(int64 position);
@ -63,6 +64,8 @@ private:
bool _mouseDown = false;
float64 _downProgress = 0.;
float64 _fadeOpacity = 1.;
};
} // namespace Clip

View File

@ -28,8 +28,10 @@ mediaviewOverDuration: 150;
mediaviewControllerSize: size(600px, 50px);
mediaviewPlayProgressLabel: LabelSimple(defaultLabelSimple) {
font: semiboldFont;
textFg: #ffffffc7;
}
mediaviewPlayProgressTop: 8px;
mediaviewPlayButton: IconButton {
width: 25px;
height: 24px;
@ -63,6 +65,7 @@ mediaviewFullScreenOutIcon: icon {
mediaviewPlaybackActive: #ffffff;
mediaviewPlaybackInactive: #474747;
mediaviewPlaybackWidth: 3px;
mediaviewPlaybackTop: 23px;
mediaviewSeekSize: size(2px, 13px);
mediaviewVolumeSize: size(44px, 12px);

View File

@ -29,6 +29,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
#include "media/media_clip_reader.h"
#include "media/view/media_clip_controller.h"
#include "styles/style_mediaview.h"
#include "media/media_audio.h"
namespace {
@ -233,6 +234,9 @@ void MediaView::stopGif() {
_gif = nullptr;
_clipController.destroy();
Sandbox::removeEventFilter(this);
if (audioPlayer()) {
disconnect(audioPlayer(), SIGNAL(updated(const AudioMsgId&)), this, SLOT(onVideoPlayProgress(const AudioMsgId&)));
}
}
void MediaView::documentUpdated(DocumentData *doc) {
@ -1247,6 +1251,10 @@ void MediaView::createClipController() {
Sandbox::removeEventFilter(this);
Sandbox::installEventFilter(this);
if (audioPlayer()) {
connect(audioPlayer(), SIGNAL(updated(const AudioMsgId&)), this, SLOT(onVideoPlayProgress(const AudioMsgId&)));
}
}
void MediaView::setClipControllerGeometry() {
@ -1289,6 +1297,19 @@ void MediaView::onVideoFromFullScreen() {
}
void MediaView::onVideoPlayProgress(const AudioMsgId &audioId) {
if (audioId.type() != AudioMsgId::Type::Video) {
return;
}
t_assert(_gif != nullptr);
t_assert(audioPlayer() != nullptr);
auto state = audioPlayer()->currentVideoState(_gif->playId());
if (state.frequency) {
_clipController->updatePlayback(state);
}
}
void MediaView::paintEvent(QPaintEvent *e) {
QRect r(e->rect());
QRegion region(e->region());

View File

@ -122,6 +122,7 @@ private slots:
void onVideoVolumeChanged(float64 volume);
void onVideoToFullScreen();
void onVideoFromFullScreen();
void onVideoPlayProgress(const AudioMsgId &audioId);
private:
void displayPhoto(PhotoData *photo, HistoryItem *item);

View File

@ -29,20 +29,26 @@ FadeAnimation::FadeAnimation(TWidget *widget) : _widget(widget) {
bool FadeAnimation::paint(Painter &p) {
if (_cache.isNull()) return false;
bool animating = _animation.animating(getms());
p.setOpacity(_animation.current(_visible ? 1. : 0.));
p.drawPixmap(0, 0, _cache);
if (!animating) {
stopAnimation();
}
return true;
}
void FadeAnimation::refreshCache() {
if (!_cache.isNull()) {
_cache = QPixmap();
_cache = myGrab(_widget);
}
}
void FadeAnimation::setFinishedCallback(FinishedCallback &&callback) {
_finishedCallback = std_::move(callback);
}
void FadeAnimation::setUpdatedCallback(UpdatedCallback &&callback) {
_updatedCallback = std_::move(callback);
}
void FadeAnimation::show() {
_visible = true;
stopAnimation();
@ -93,7 +99,12 @@ void FadeAnimation::startAnimation(int duration) {
}
void FadeAnimation::updateCallback() {
_widget->update();
if (_animation.animating(getms())) {
_widget->update();
_updatedCallback.call(_animation.current());
} else {
stopAnimation();
}
}
} // namespace Ui

View File

@ -29,10 +29,14 @@ public:
FadeAnimation(TWidget *widget);
bool paint(Painter &p);
void refreshCache();
using FinishedCallback = Function<void>;
void setFinishedCallback(FinishedCallback &&callback);
using UpdatedCallback = Function<void, float64>;
void setUpdatedCallback(UpdatedCallback &&callback);
void show();
void hide();
@ -51,6 +55,7 @@ private:
bool _visible = false;
FinishedCallback _finishedCallback;
UpdatedCallback _updatedCallback;
};

View File

@ -28,17 +28,29 @@ LabelSimple::LabelSimple(QWidget *parent, const style::LabelSimple &st, const QS
setText(value);
}
void LabelSimple::setText(const QString &value) {
void LabelSimple::setText(const QString &value, bool *outTextChanged) {
if (_fullText == value) {
if (outTextChanged) *outTextChanged = false;
return;
}
_fullText = value;
_fullTextWidth = _st.font->width(_fullText);
if (!_st.maxWidth || _fullTextWidth <= _st.maxWidth) {
_text = _fullText;
_textWidth = _fullTextWidth;
} else {
_text = _st.font->elided(_fullText, _st.maxWidth);
auto newText = _st.font->elided(_fullText, _st.maxWidth);
if (newText == _text) {
if (outTextChanged) *outTextChanged = false;
return;
}
_text = newText;
_textWidth = _st.font->width(_text);
}
resize(_textWidth, _st.font->height);
update();
if (outTextChanged) *outTextChanged = true;
}
void LabelSimple::paintEvent(QPaintEvent *e) {

View File

@ -29,7 +29,7 @@ public:
LabelSimple(QWidget *parent, const style::LabelSimple &st = st::defaultLabelSimple, const QString &value = QString());
// This method also resizes the label.
void setText(const QString &newText);
void setText(const QString &newText, bool *outTextChanged = nullptr);
protected:
void paintEvent(QPaintEvent *e) override;