mirror of https://github.com/procxx/kepka.git
Optimize locked videos repainting.
This commit is contained in:
parent
40d4353d05
commit
92d87f96e1
|
@ -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() {
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue