Optimize locked videos repainting.

This commit is contained in:
John Preston 2019-12-18 21:48:07 +03:00
parent 40d4353d05
commit 92d87f96e1
2 changed files with 75 additions and 56 deletions

View File

@ -47,6 +47,22 @@ int gifMaxStatusWidth(DocumentData *document) {
} // namespace } // namespace
struct Gif::Streamed {
Streamed(
std::shared_ptr<::Media::Streaming::Document> shared,
Fn<void()> waitingCallback);
::Media::Streaming::Instance instance;
::Media::Streaming::FrameRequest frozenRequest;
QImage frozenFrame;
QString frozenStatusText;
};
Gif::Streamed::Streamed(
std::shared_ptr<::Media::Streaming::Document> shared,
Fn<void()> waitingCallback)
: instance(std::move(shared), std::move(waitingCallback)) {
}
Gif::Gif( Gif::Gif(
not_null<Element*> parent, not_null<Element*> parent,
not_null<HistoryItem*> realParent, not_null<HistoryItem*> realParent,
@ -237,6 +253,7 @@ void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms
const auto cornerDownload = downloadInCorner(); const auto cornerDownload = downloadInCorner();
const auto canBePlayed = _data->canBePlayed(); const auto canBePlayed = _data->canBePlayed();
const auto activeRoundPlaying = activeRoundStreamed(); const auto activeRoundPlaying = activeRoundStreamed();
const auto activeOwnPlaying = activeOwnStreamed();
const auto autoplay = autoplayEnabled() && canBePlayed; const auto autoplay = autoplayEnabled() && canBePlayed;
const auto streamingMode = _streamed || activeRoundPlaying || autoplay; const auto streamingMode = _streamed || activeRoundPlaying || autoplay;
const auto startPlayAsync = autoplay const auto startPlayAsync = autoplay
@ -246,7 +263,9 @@ void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms
if (!autoPaused) { if (!autoPaused) {
_parent->delegate()->elementAnimationAutoplayAsync(_parent); _parent->delegate()->elementAnimationAutoplayAsync(_parent);
} }
} else if (_streamed && !_streamed->active() && !_streamed->failed()) { } else if (_streamed
&& !_streamed->instance.active()
&& !_streamed->instance.failed()) {
startStreamedPlayer(); startStreamedPlayer();
} }
@ -259,7 +278,11 @@ void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms
const auto isRound = _data->isVideoMessage(); const auto isRound = _data->isVideoMessage();
auto displayMute = false; auto displayMute = false;
const auto streamed = activeCurrentStreamed(); const auto streamed = activeRoundPlaying
? activeRoundPlaying
: activeOwnPlaying
? &activeOwnPlaying->instance
: nullptr;
if ((!streamed || item->isSending()) && displayLoading) { if ((!streamed || item->isSending()) && displayLoading) {
ensureAnimation(); ensureAnimation();
@ -314,14 +337,21 @@ void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms
request.resize = QSize(_thumbw, _thumbh) * cIntRetinaFactor(); request.resize = QSize(_thumbw, _thumbh) * cIntRetinaFactor();
request.corners = roundCorners; request.corners = roundCorners;
request.radius = roundRadius; request.radius = roundRadius;
if (streamed->playerLocked() && !activeRoundPlaying) { if (!activeRoundPlaying && activeOwnPlaying->instance.playerLocked()) {
if (_lockedFrameRequest != request || _lockedFrame.isNull()) { if (activeOwnPlaying->frozenFrame.isNull()) {
_lockedFrameRequest = request; activeOwnPlaying->frozenRequest = request;
_lockedFrame = streamed->frame(request); activeOwnPlaying->frozenFrame = streamed->frame(request);
activeOwnPlaying->frozenStatusText = _statusText;
} else if (activeOwnPlaying->frozenRequest != request) {
activeOwnPlaying->frozenRequest = request;
activeOwnPlaying->frozenFrame = streamed->frame(request);
} }
p.drawImage(rthumb, _lockedFrame); p.drawImage(rthumb, activeOwnPlaying->frozenFrame);
} else { } else {
_lockedFrame = QImage(); if (activeOwnPlaying) {
activeOwnPlaying->frozenFrame = QImage();
activeOwnPlaying->frozenStatusText = QString();
}
p.drawImage(rthumb, streamed->frame(request)); p.drawImage(rthumb, streamed->frame(request));
if (!paused) { if (!paused) {
streamed->markFrameShown(); streamed->markFrameShown();
@ -579,6 +609,10 @@ void Gif::drawCornerStatus(Painter &p, bool selected) const {
if (!needInfoDisplay()) { if (!needInfoDisplay()) {
return; return;
} }
const auto own = activeOwnStreamed();
const auto text = (own && !own->frozenStatusText.isEmpty())
? own->frozenStatusText
: _statusText;
const auto padding = st::msgDateImgPadding; const auto padding = st::msgDateImgPadding;
const auto radial = _animation && _animation->radial.animating(); const auto radial = _animation && _animation->radial.animating();
const auto cornerMute = _streamed && _data->isVideoFile(); const auto cornerMute = _streamed && _data->isVideoFile();
@ -586,7 +620,7 @@ void Gif::drawCornerStatus(Painter &p, bool selected) const {
const auto addLeft = cornerDownload ? (st::historyVideoDownloadSize + 2 * padding.y()) : 0; const auto addLeft = cornerDownload ? (st::historyVideoDownloadSize + 2 * padding.y()) : 0;
const auto addRight = cornerMute ? st::historyVideoMuteSize : 0; const auto addRight = cornerMute ? st::historyVideoMuteSize : 0;
const auto downloadWidth = cornerDownload ? st::normalFont->width(_downloadSize) : 0; const auto downloadWidth = cornerDownload ? st::normalFont->width(_downloadSize) : 0;
const auto statusW = std::max(downloadWidth, st::normalFont->width(_statusText)) + 2 * padding.x() + addLeft + addRight; const auto statusW = std::max(downloadWidth, st::normalFont->width(text)) + 2 * padding.x() + addLeft + addRight;
const auto statusH = cornerDownload ? (st::historyVideoDownloadSize + 2 * padding.y()) : (st::normalFont->height + 2 * padding.y()); const auto statusH = cornerDownload ? (st::historyVideoDownloadSize + 2 * padding.y()) : (st::normalFont->height + 2 * padding.y());
const auto statusX = st::msgDateImgDelta + padding.x(); const auto statusX = st::msgDateImgDelta + padding.x();
const auto statusY = st::msgDateImgDelta + padding.y(); const auto statusY = st::msgDateImgDelta + padding.y();
@ -595,7 +629,7 @@ void Gif::drawCornerStatus(Painter &p, bool selected) const {
App::roundRect(p, around, selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners); App::roundRect(p, around, selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners);
p.setFont(st::normalFont); p.setFont(st::normalFont);
p.setPen(st::msgDateImgFg); p.setPen(st::msgDateImgFg);
p.drawTextLeft(statusX + addLeft, statusTextTop, width(), _statusText, statusW - 2 * padding.x()); p.drawTextLeft(statusX + addLeft, statusTextTop, width(), text, statusW - 2 * padding.x());
if (cornerDownload) { if (cornerDownload) {
const auto downloadTextTop = statusY + st::normalFont->height + (2 * (statusH - 2 * st::normalFont->height) / 3) - padding.y(); const auto downloadTextTop = statusY + st::normalFont->height + (2 * (statusH - 2 * st::normalFont->height) / 3) - padding.y();
p.drawTextLeft(statusX + addLeft, downloadTextTop, width(), _downloadSize, statusW - 2 * padding.x()); p.drawTextLeft(statusX + addLeft, downloadTextTop, width(), _downloadSize, statusW - 2 * padding.x());
@ -1042,8 +1076,8 @@ void Gif::updateStatusText() const {
} }
const auto round = activeRoundStreamed(); const auto round = activeRoundStreamed();
const auto own = activeOwnStreamed(); const auto own = activeOwnStreamed();
if (round || (own && _data->isVideoFile())) { if (round || (own && own->frozenFrame.isNull() && _data->isVideoFile())) {
const auto streamed = round ? round : own; const auto streamed = round ? round : &own->instance;
const auto state = streamed->player().prepareLegacyState(); const auto state = streamed->player().prepareLegacyState();
if (state.length) { if (state.length) {
auto position = int64(0); auto position = int64(0);
@ -1098,10 +1132,10 @@ int Gif::additionalWidth(const HistoryMessageVia *via, const HistoryMessageReply
return ::Media::Player::instance()->roundVideoStreamed(_parent->data()); return ::Media::Player::instance()->roundVideoStreamed(_parent->data());
} }
::Media::Streaming::Instance *Gif::activeOwnStreamed() const { Gif::Streamed *Gif::activeOwnStreamed() const {
return (_streamed return (_streamed
&& _streamed->player().ready() && _streamed->instance.player().ready()
&& !_streamed->player().videoSize().isEmpty()) && !_streamed->instance.player().videoSize().isEmpty())
? _streamed.get() ? _streamed.get()
: nullptr; : nullptr;
} }
@ -1109,8 +1143,10 @@ int Gif::additionalWidth(const HistoryMessageVia *via, const HistoryMessageReply
::Media::Streaming::Instance *Gif::activeCurrentStreamed() const { ::Media::Streaming::Instance *Gif::activeCurrentStreamed() const {
if (const auto streamed = activeRoundStreamed()) { if (const auto streamed = activeRoundStreamed()) {
return streamed; return streamed;
} else if (const auto owned = activeOwnStreamed()) {
return &owned->instance;
} }
return activeOwnStreamed(); return nullptr;
} }
::Media::View::PlaybackProgress *Gif::videoPlayback() const { ::Media::View::PlaybackProgress *Gif::videoPlayback() const {
@ -1144,19 +1180,19 @@ void Gif::createStreamedPlayer() {
if (!shared) { if (!shared) {
return; return;
} }
setStreamed(std::make_unique<::Media::Streaming::Instance>( setStreamed(std::make_unique<Streamed>(
std::move(shared), std::move(shared),
[=] { history()->owner().requestViewRepaint(_parent); })); [=] { repaintStreamedContent(); }));
_streamed->player().updates( _streamed->instance.player().updates(
) | rpl::start_with_next_error([=](::Media::Streaming::Update &&update) { ) | rpl::start_with_next_error([=](::Media::Streaming::Update &&update) {
handleStreamingUpdate(std::move(update)); handleStreamingUpdate(std::move(update));
}, [=](::Media::Streaming::Error &&error) { }, [=](::Media::Streaming::Error &&error) {
handleStreamingError(std::move(error)); handleStreamingError(std::move(error));
}, _streamed->lifetime()); }, _streamed->instance.lifetime());
if (_streamed->ready()) { if (_streamed->instance.ready()) {
streamingReady(base::duplicate(_streamed->info())); streamingReady(base::duplicate(_streamed->instance.info()));
} }
startStreamedPlayer(); startStreamedPlayer();
} }
@ -1171,10 +1207,10 @@ void Gif::startStreamedPlayer() const {
options.mode = ::Media::Streaming::Mode::Video; options.mode = ::Media::Streaming::Mode::Video;
options.loop = true; options.loop = true;
//} //}
_streamed->play(options); _streamed->instance.play(options);
} }
void Gif::setStreamed(std::unique_ptr<::Media::Streaming::Instance> value) { void Gif::setStreamed(std::unique_ptr<Streamed> value) {
const auto removed = (_streamed && !value); const auto removed = (_streamed && !value);
const auto set = (!_streamed && value); const auto set = (!_streamed && value);
if (removed) { if (removed) {
@ -1192,47 +1228,29 @@ void Gif::handleStreamingUpdate(::Media::Streaming::Update &&update) {
update.data.match([&](Information &update) { update.data.match([&](Information &update) {
streamingReady(std::move(update)); streamingReady(std::move(update));
}, [&](const PreloadedVideo &update) { }, [&](const PreloadedVideo &update) {
//updatePlaybackState();
}, [&](const UpdateVideo &update) { }, [&](const UpdateVideo &update) {
history()->owner().requestViewRepaint(_parent); repaintStreamedContent();
//updatePlaybackState();
}, [&](const PreloadedAudio &update) { }, [&](const PreloadedAudio &update) {
//_streamed->info.audio.state.receivedTill = update.till;
//updatePlaybackState();
}, [&](const UpdateAudio &update) { }, [&](const UpdateAudio &update) {
//_streamed->info.audio.state.position = update.position;
//updatePlaybackState();
}, [&](const WaitingForData &update) { }, [&](const WaitingForData &update) {
//playbackWaitingChange(update.waiting);
}, [&](MutedByOther) { }, [&](MutedByOther) {
}, [&](Finished) { }, [&](Finished) {
//updatePlaybackState();
}); });
} }
void Gif::handleStreamingError(::Media::Streaming::Error &&error) { void Gif::handleStreamingError(::Media::Streaming::Error &&error) {
//if (!_data->canBePlayed()) { }
// redisplayContent();
//} else { void Gif::repaintStreamedContent() {
// playbackWaitingChange(false); const auto own = activeOwnStreamed();
// updatePlaybackState(); if (own && !own->frozenFrame.isNull()) {
//} return;
}
history()->owner().requestViewRepaint(_parent);
} }
void Gif::streamingReady(::Media::Streaming::Information &&info) { void Gif::streamingReady(::Media::Streaming::Information &&info) {
history()->owner().requestViewResize(_parent); history()->owner().requestViewResize(_parent);
//validateStreamedGoodThumbnail();
//if (videoShown()) {
// const auto contentSize = ConvertScale(videoSize());
// if (contentSize != QSize(_width, _height)) {
// update(contentRect());
// _w = contentSize.width();
// _h = contentSize.height();
// contentSizeChanged();
// }
//}
//history()->owner().requestViewRepaint(_parent);
//playbackWaitingChange(false);
} }
void Gif::stopAnimation() { void Gif::stopAnimation() {

View File

@ -102,6 +102,8 @@ public:
} }
private: private:
struct Streamed;
float64 dataProgress() const override; float64 dataProgress() const override;
bool dataFinished() const override; bool dataFinished() const override;
bool dataLoaded() const override; bool dataLoaded() const override;
@ -113,16 +115,17 @@ private:
QSize countCurrentSize(int newWidth) override; QSize countCurrentSize(int newWidth) override;
QSize videoSize() const; QSize videoSize() const;
::Media::Streaming::Instance *activeRoundStreamed() const; ::Media::Streaming::Instance *activeRoundStreamed() const;
::Media::Streaming::Instance *activeOwnStreamed() const; Streamed *activeOwnStreamed() const;
::Media::Streaming::Instance *activeCurrentStreamed() const; ::Media::Streaming::Instance *activeCurrentStreamed() const;
::Media::View::PlaybackProgress *videoPlayback() const; ::Media::View::PlaybackProgress *videoPlayback() const;
void createStreamedPlayer(); void createStreamedPlayer();
void startStreamedPlayer() const; void startStreamedPlayer() const;
void setStreamed(std::unique_ptr<::Media::Streaming::Instance> value); void setStreamed(std::unique_ptr<Streamed> value);
void handleStreamingUpdate(::Media::Streaming::Update &&update); void handleStreamingUpdate(::Media::Streaming::Update &&update);
void handleStreamingError(::Media::Streaming::Error &&error); void handleStreamingError(::Media::Streaming::Error &&error);
void streamingReady(::Media::Streaming::Information &&info); void streamingReady(::Media::Streaming::Information &&info);
void repaintStreamedContent();
bool needInfoDisplay() const; bool needInfoDisplay() const;
int additionalWidth( int additionalWidth(
@ -152,10 +155,8 @@ private:
int _thumbw = 1; int _thumbw = 1;
int _thumbh = 1; int _thumbh = 1;
Ui::Text::String _caption; Ui::Text::String _caption;
std::unique_ptr<::Media::Streaming::Instance> _streamed; std::unique_ptr<Streamed> _streamed;
mutable ::Media::Streaming::FrameRequest _lockedFrameRequest;
mutable QImage _lockedFrame;
QString _downloadSize; QString _downloadSize;
}; };