diff --git a/Telegram/SourceFiles/boxes/sticker_set_box.cpp b/Telegram/SourceFiles/boxes/sticker_set_box.cpp index 8310f7306..5324b3e61 100644 --- a/Telegram/SourceFiles/boxes/sticker_set_box.cpp +++ b/Telegram/SourceFiles/boxes/sticker_set_box.cpp @@ -21,7 +21,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "ui/image/image.h" #include "ui/text/text_utilities.h" #include "ui/emoji_config.h" -#include "lottie/lottie_single_player.h" +#include "lottie/lottie_multi_player.h" +#include "lottie/lottie_animation.h" #include "window/window_session_controller.h" #include "auth_session.h" #include "apiwrap.h" @@ -65,10 +66,12 @@ protected: private: struct Element { not_null document; - std::unique_ptr animated; + Lottie::Animation *animated = nullptr; Ui::Animations::Simple overAnimation; }; + void visibleTopBottomUpdated(int visibleTop, int visibleBottom) override; + QSize boundingBoxSize() const; void paintSticker(Painter &p, int index, QPoint position) const; @@ -86,11 +89,14 @@ private: return (_setFlags & MTPDstickerSet::Flag::f_masks); } + not_null getLottiePlayer(); + void showPreview(); not_null _controller; MTP::Sender _mtp; std::vector _elements; + std::unique_ptr _lottiePlayer; Stickers::Pack _pack; Stickers::ByEmojiMap _emoji; bool _loaded = false; @@ -472,6 +478,18 @@ void StickerSetBox::Inner::showPreview() { } } +not_null StickerSetBox::Inner::getLottiePlayer() { + if (!_lottiePlayer) { + _lottiePlayer = std::make_unique( + Lottie::MakeFrameRenderer()); + _lottiePlayer->updates( + ) | rpl::start_with_next([=] { + update(); + }, lifetime()); + } + return _lottiePlayer.get(); +} + int32 StickerSetBox::Inner::stickerFromGlobalPos(const QPoint &p) const { QPoint l(mapFromGlobal(p)); if (rtl()) l.setX(width() - l.x()); @@ -492,7 +510,8 @@ void StickerSetBox::Inner::paintEvent(QPaintEvent *e) { return; } - int32 rows = _elements.size() / kStickersPanelPerRow + ((_elements.size() % kStickersPanelPerRow) ? 1 : 0); + int32 rows = (_elements.size() / kStickersPanelPerRow) + + ((_elements.size() % kStickersPanelPerRow) ? 1 : 0); int32 from = qFloor(e->rect().top() / st::stickersSize.height()), to = qFloor(e->rect().bottom() / st::stickersSize.height()) + 1; for (int32 i = from; i < to; ++i) { @@ -505,6 +524,14 @@ void StickerSetBox::Inner::paintEvent(QPaintEvent *e) { paintSticker(p, index, pos); } } + + if (_lottiePlayer) { + const auto paused = _controller->isGifPausedAtLeastFor( + Window::GifPauseReason::Layer); + if (!paused) { + _lottiePlayer->markFrameShown(); + } + } } QSize StickerSetBox::Inner::boundingBoxSize() const { @@ -513,20 +540,56 @@ QSize StickerSetBox::Inner::boundingBoxSize() const { st::stickersSize.height() - st::buttonRadius * 2); } +void StickerSetBox::Inner::visibleTopBottomUpdated( + int visibleTop, + int visibleBottom) { + const auto pauseInRows = [&](int fromRow, int tillRow) { + Expects(fromRow <= tillRow); + + for (auto i = fromRow; i != tillRow; ++i) { + for (auto j = 0; j != kStickersPanelPerRow; ++j) { + const auto index = i * kStickersPanelPerRow + j; + if (index >= _elements.size()) { + break; + } + if (const auto animated = _elements[index].animated) { + _lottiePlayer->pause(animated); + } + } + } + }; + const auto count = int(_elements.size()); + const auto rowsCount = (count / kStickersPanelPerRow) + + ((count % kStickersPanelPerRow) ? 1 : 0); + const auto rowsTop = st::stickersPadding.top(); + const auto singleHeight = st::stickersSize.height(); + const auto rowsBottom = rowsTop + rowsCount * singleHeight; + if (visibleTop >= rowsTop + singleHeight && visibleTop < rowsBottom) { + const auto pauseHeight = (visibleTop - rowsTop); + const auto pauseRows = std::min( + pauseHeight / singleHeight, + rowsCount); + pauseInRows(0, pauseRows); + } + if (visibleBottom > rowsTop + && visibleBottom + singleHeight <= rowsBottom) { + const auto pauseHeight = (rowsBottom - visibleBottom); + const auto pauseRows = std::min( + pauseHeight / singleHeight, + rowsCount); + pauseInRows(rowsCount - pauseRows, rowsCount); + } +} + void StickerSetBox::Inner::setupLottie(int index) { auto &element = _elements[index]; const auto document = element.document; - element.animated = Stickers::LottiePlayerFromDocument( + element.animated = Stickers::LottieAnimationFromDocument( + getLottiePlayer(), document, Stickers::LottieSize::StickerSet, boundingBoxSize() * cIntRetinaFactor()); - const auto animation = element.animated.get(); - - animation->updates( - ) | rpl::start_with_next([=] { - update(); - }, lifetime()); } void StickerSetBox::Inner::paintSticker( @@ -558,15 +621,12 @@ void StickerSetBox::Inner::paintSticker( if (h < 1) h = 1; QPoint ppos = position + QPoint((st::stickersSize.width() - w) / 2, (st::stickersSize.height() - h) / 2); if (element.animated && element.animated->ready()) { - const auto paused = _controller->isGifPausedAtLeastFor( - Window::GifPauseReason::Layer); - if (!paused) { - element.animated->markFrameShown(); - } const auto frame = element.animated->frame(); p.drawImage( QRect(ppos, frame.size() / cIntRetinaFactor()), frame); + + _lottiePlayer->unpause(element.animated); } else if (const auto image = document->getStickerSmall()) { p.drawPixmapLeft( ppos, diff --git a/Telegram/SourceFiles/lottie/lottie_multi_player.cpp b/Telegram/SourceFiles/lottie/lottie_multi_player.cpp index 3620d1417..ae8aba129 100644 --- a/Telegram/SourceFiles/lottie/lottie_multi_player.cpp +++ b/Telegram/SourceFiles/lottie/lottie_multi_player.cpp @@ -98,7 +98,7 @@ void MultiPlayer::start( if (_active.empty() || (_lastSyncTime == kTimeUnknown && _nextFrameTime == kTimeUnknown)) { - startBeforeLifeCycle(animation, std::move(info)); + addNewToActive(animation, std::move(info)); } else { // We always try to mark as shown at the same time, so we start a new // animation at the same time we mark all existing as shown. @@ -107,7 +107,7 @@ void MultiPlayer::start( _updates.fire({}); } -void MultiPlayer::startBeforeLifeCycle( +void MultiPlayer::addNewToActive( not_null animation, StartingInfo &&info) { _active.emplace(animation, info.state.get()); @@ -117,20 +117,6 @@ void MultiPlayer::startBeforeLifeCycle( } } -void MultiPlayer::startInsideLifeCycle( - not_null animation, - StartingInfo &&info) { - const auto state = info.state.get(); - if (info.paused) { - _paused.emplace( - animation, - PausedInfo{ state, _lastSyncTime, _delay }); - } else { - _active.emplace(animation, state); - } - startAtRightTime(std::move(info.state)); -} - void MultiPlayer::processPending() { Expects(_lastSyncTime != kTimeUnknown); @@ -141,7 +127,7 @@ void MultiPlayer::processPending() { unpauseAndKeepUp(animation); } for (auto &[animation, info] : base::take(_pendingToStart)) { - startInsideLifeCycle(animation, std::move(info)); + addNewToActive(animation, std::move(info)); } for (const auto &animation : base::take(_pendingRemove)) { removeNow(animation); diff --git a/Telegram/SourceFiles/lottie/lottie_multi_player.h b/Telegram/SourceFiles/lottie/lottie_multi_player.h index 849f28cb8..387316ccc 100644 --- a/Telegram/SourceFiles/lottie/lottie_multi_player.h +++ b/Telegram/SourceFiles/lottie/lottie_multi_player.h @@ -69,10 +69,7 @@ private: bool paused = false; }; - void startBeforeLifeCycle( - not_null animation, - StartingInfo &&info); - void startInsideLifeCycle( + void addNewToActive( not_null animation, StartingInfo &&info); [[nodiscard]] int countFrameIndex(