Cache last frame of stickers panel footer icons.

This commit is contained in:
John Preston 2020-05-21 17:56:59 +04:00
parent ad5507f2c8
commit eb75859dc0
3 changed files with 87 additions and 71 deletions

View File

@ -81,12 +81,14 @@ struct StickerIcon {
uint64 setId = 0; uint64 setId = 0;
ImagePtr thumbnail; ImagePtr thumbnail;
mutable Lottie::SinglePlayer *lottie = nullptr; mutable std::unique_ptr<Lottie::SinglePlayer> lottie;
mutable QPixmap savedFrame;
DocumentData *sticker = nullptr; DocumentData *sticker = nullptr;
mutable std::shared_ptr<Data::DocumentMedia> stickerMedia; mutable std::shared_ptr<Data::DocumentMedia> stickerMedia;
ChannelData *megagroup = nullptr; ChannelData *megagroup = nullptr;
int pixw = 0; int pixw = 0;
int pixh = 0; int pixh = 0;
mutable rpl::lifetime lifetime;
}; };
@ -109,7 +111,7 @@ public:
void returnFocus(); void returnFocus();
void setLoading(bool loading); void setLoading(bool loading);
void clearLottieData(); void clearHeavyData();
protected: protected:
void paintEvent(QPaintEvent *e) override; void paintEvent(QPaintEvent *e) override;
@ -129,12 +131,6 @@ private:
}; };
using OverState = base::variant<SpecialOver, int>; using OverState = base::variant<SpecialOver, int>;
struct LottieIcon {
std::unique_ptr<Lottie::SinglePlayer> player;
bool stale = false;
rpl::lifetime lifetime;
};
template <typename Callback> template <typename Callback>
void enumerateVisibleIcons(Callback callback); void enumerateVisibleIcons(Callback callback);
@ -145,7 +141,6 @@ private:
void validateIconLottieAnimation(const StickerIcon &icon); void validateIconLottieAnimation(const StickerIcon &icon);
void refreshIconsGeometry(ValidateIconAnimations animations); void refreshIconsGeometry(ValidateIconAnimations animations);
void refillLottieData();
void updateSelected(); void updateSelected();
void updateSetIcon(uint64 setId); void updateSetIcon(uint64 setId);
void finishDragging(); void finishDragging();
@ -164,8 +159,7 @@ private:
static constexpr auto kVisibleIconsCount = 8; static constexpr auto kVisibleIconsCount = 8;
QList<StickerIcon> _icons; std::vector<StickerIcon> _icons;
mutable base::flat_map<uint64, LottieIcon> _lottieData;
OverState _iconOver = SpecialOver::None; OverState _iconOver = SpecialOver::None;
int _iconSel = 0; int _iconSel = 0;
OverState _iconDown = SpecialOver::None; OverState _iconDown = SpecialOver::None;
@ -238,30 +232,25 @@ StickersListWidget::Footer::Footer(not_null<StickersListWidget*> parent)
}); });
} }
void StickersListWidget::Footer::clearLottieData() { void StickersListWidget::Footer::clearHeavyData() {
for (auto &icon : _icons) { const auto count = int(_icons.size());
const auto iconsX = qRound(_iconsX.current());
const auto index = iconsX / st::stickerIconWidth;
const auto x = _iconsLeft - (iconsX % st::stickerIconWidth);
const auto first = floorclamp(iconsX, st::stickerIconWidth, 0, count);
const auto last = ceilclamp(
iconsX + width(),
st::stickerIconWidth,
0,
count);
for (auto i = 0; i != count; ++i) {
auto &icon = _icons[i];
icon.lottie = nullptr; icon.lottie = nullptr;
} icon.lifetime.destroy();
_lottieData.clear(); icon.stickerMedia = nullptr;
} if (i < first || i >= last) {
// Not visible icon.
void StickersListWidget::Footer::refillLottieData() { icon.savedFrame = QPixmap();
for (auto &item : _lottieData) {
item.second.stale = true;
}
for (auto &icon : _icons) {
const auto i = _lottieData.find(icon.setId);
if (i == end(_lottieData)) {
continue;
}
icon.lottie = i->second.player.get();
i->second.stale = false;
}
for (auto i = begin(_lottieData); i != end(_lottieData);) {
if (i->second.stale) {
i = _lottieData.erase(i);
} else {
++i;
} }
} }
} }
@ -360,7 +349,7 @@ void StickersListWidget::Footer::validateSelectedIcon(
ValidateIconAnimations animations) { ValidateIconAnimations animations) {
auto favedIconIndex = -1; auto favedIconIndex = -1;
auto newSelected = -1; auto newSelected = -1;
for (auto i = 0, l = _icons.size(); i != l; ++i) { for (auto i = 0, l = int(_icons.size()); i != l; ++i) {
if (_icons[i].setId == setId if (_icons[i].setId == setId
|| (_icons[i].setId == Stickers::FavedSetId || (_icons[i].setId == Stickers::FavedSetId
&& setId == Stickers::RecentSetId)) { && setId == Stickers::RecentSetId)) {
@ -434,7 +423,7 @@ void StickersListWidget::Footer::paintEvent(QPaintEvent *e) {
paintSearchIcon(p); paintSearchIcon(p);
if (_icons.isEmpty() || _searchShown) { if (_icons.empty() || _searchShown) {
return; return;
} }
@ -536,7 +525,7 @@ void StickersListWidget::Footer::mouseMoveEvent(QMouseEvent *e) {
updateSelected(); updateSelected();
if (!_iconsDragging if (!_iconsDragging
&& !_icons.isEmpty() && !_icons.empty()
&& base::get_if<int>(&_iconDown) != nullptr) { && base::get_if<int>(&_iconDown) != nullptr) {
if ((_iconsMousePos - _iconsMouseDown).manhattanLength() >= QApplication::startDragDistance()) { if ((_iconsMousePos - _iconsMouseDown).manhattanLength() >= QApplication::startDragDistance()) {
_iconsDragging = true; _iconsDragging = true;
@ -554,7 +543,7 @@ void StickersListWidget::Footer::mouseMoveEvent(QMouseEvent *e) {
} }
void StickersListWidget::Footer::mouseReleaseEvent(QMouseEvent *e) { void StickersListWidget::Footer::mouseReleaseEvent(QMouseEvent *e) {
if (_icons.isEmpty()) { if (_icons.empty()) {
return; return;
} }
@ -592,7 +581,7 @@ void StickersListWidget::Footer::finishDragging() {
bool StickersListWidget::Footer::event(QEvent *e) { bool StickersListWidget::Footer::event(QEvent *e) {
if (e->type() == QEvent::TouchBegin) { if (e->type() == QEvent::TouchBegin) {
} else if (e->type() == QEvent::Wheel) { } else if (e->type() == QEvent::Wheel) {
if (!_icons.isEmpty() if (!_icons.empty()
&& (base::get_if<int>(&_iconOver) != nullptr) && (base::get_if<int>(&_iconOver) != nullptr)
&& (_iconDown == SpecialOver::None)) { && (_iconDown == SpecialOver::None)) {
scrollByWheelEvent(static_cast<QWheelEvent*>(e)); scrollByWheelEvent(static_cast<QWheelEvent*>(e));
@ -643,10 +632,10 @@ void StickersListWidget::Footer::updateSelected() {
&& x < settingsLeft + st::stickerIconWidth && x < settingsLeft + st::stickerIconWidth
&& y >= _iconsTop && y >= _iconsTop
&& y < _iconsTop + st::emojiFooterHeight) { && y < _iconsTop + st::emojiFooterHeight) {
if (!_icons.isEmpty() && !hasOnlyFeaturedSets()) { if (!_icons.empty() && !hasOnlyFeaturedSets()) {
newOver = SpecialOver::Settings; newOver = SpecialOver::Settings;
} }
} else if (!_icons.isEmpty()) { } else if (!_icons.empty()) {
if (y >= _iconsTop if (y >= _iconsTop
&& y < _iconsTop + st::emojiFooterHeight && y < _iconsTop + st::emojiFooterHeight
&& x >= _iconsLeft && x >= _iconsLeft
@ -669,8 +658,28 @@ void StickersListWidget::Footer::updateSelected() {
void StickersListWidget::Footer::refreshIcons( void StickersListWidget::Footer::refreshIcons(
ValidateIconAnimations animations) { ValidateIconAnimations animations) {
_pan->fillIcons(_icons); auto icons = _pan->fillIcons();
refillLottieData();
auto indices = base::flat_map<uint64, int>();
indices.reserve(_icons.size());
auto index = 0;
for (const auto &entry : _icons) {
indices.emplace(entry.setId, index++);
}
for (auto &now : icons) {
if (const auto i = indices.find(now.setId); i != end(indices)) {
auto &was = _icons[i->second];
if (now.sticker == was.sticker
&& now.thumbnail.get() == was.thumbnail.get()) {
now.lottie = std::move(was.lottie);
now.lifetime = std::move(was.lifetime);
now.savedFrame = std::move(was.savedFrame);
}
}
}
_icons = std::move(icons);
refreshIconsGeometry(animations); refreshIconsGeometry(animations);
} }
@ -727,17 +736,13 @@ void StickersListWidget::Footer::validateIconLottieAnimation(
if (!player) { if (!player) {
return; return;
} }
icon.lottie = player.get(); icon.lottie = std::move(player);
const auto id = icon.setId;
const auto [i, ok] = _lottieData.emplace(
id,
LottieIcon{ std::move(player) });
Assert(ok);
const auto id = icon.setId;
icon.lottie->updates( icon.lottie->updates(
) | rpl::start_with_next([=] { ) | rpl::start_with_next([=] {
updateSetIcon(id); updateSetIcon(id);
}, i->second.lifetime); }, icon.lifetime);
} }
void StickersListWidget::Footer::updateSetIcon(uint64 setId) { void StickersListWidget::Footer::updateSetIcon(uint64 setId) {
@ -759,23 +764,33 @@ void StickersListWidget::Footer::paintSetIcon(
const auto thumb = icon.thumbnail const auto thumb = icon.thumbnail
? icon.thumbnail.get() ? icon.thumbnail.get()
: icon.stickerMedia->thumbnail(); : icon.stickerMedia->thumbnail();
if (!thumb) { if (thumb) {
return; thumb->load(origin);
} }
thumb->load(origin);
const_cast<Footer*>(this)->validateIconLottieAnimation(icon); const_cast<Footer*>(this)->validateIconLottieAnimation(icon);
if (!icon.lottie) { if (!icon.lottie
if (!thumb->loaded()) { || (!icon.lottie->ready() && !icon.savedFrame.isNull())) {
const auto pixmap = !icon.savedFrame.isNull()
? icon.savedFrame
: (!icon.lottie && thumb && thumb->loaded())
? thumb->pix(origin, icon.pixw, icon.pixh)
: QPixmap();
if (pixmap.isNull()) {
return; return;
} else if (icon.savedFrame.isNull()) {
icon.savedFrame = pixmap;
} }
p.drawPixmapLeft( p.drawPixmapLeft(
x + (st::stickerIconWidth - icon.pixw) / 2, x + (st::stickerIconWidth - icon.pixw) / 2,
_iconsTop + (st::emojiFooterHeight - icon.pixh) / 2, _iconsTop + (st::emojiFooterHeight - icon.pixh) / 2,
width(), width(),
thumb->pix(origin, icon.pixw, icon.pixh)); pixmap);
} else if (icon.lottie->ready()) { } else if (icon.lottie->ready()) {
const auto frame = icon.lottie->frame(); const auto frame = icon.lottie->frame();
const auto size = frame.size() / cIntRetinaFactor(); const auto size = frame.size() / cIntRetinaFactor();
if (icon.savedFrame.isNull()) {
icon.savedFrame = QPixmap::fromImage(frame, Qt::ColorOnly);
}
p.drawImage( p.drawImage(
QRect( QRect(
x + (st::stickerIconWidth - size.width()) / 2, x + (st::stickerIconWidth - size.width()) / 2,
@ -2117,7 +2132,7 @@ void StickersListWidget::processHideFinished() {
clearSelection(); clearSelection();
clearLottieData(); clearLottieData();
if (_footer) { if (_footer) {
_footer->clearLottieData(); _footer->clearHeavyData();
} }
} }
@ -2125,7 +2140,7 @@ void StickersListWidget::processPanelHideFinished() {
clearInstalledLocally(); clearInstalledLocally();
clearLottieData(); clearLottieData();
if (_footer) { if (_footer) {
_footer->clearLottieData(); _footer->clearHeavyData();
} }
// Preserve panel state through visibility toggles. // Preserve panel state through visibility toggles.
//// Reset to the recent stickers section. //// Reset to the recent stickers section.
@ -2581,28 +2596,28 @@ void StickersListWidget::refreshMegagroupStickers(GroupStickersPlace place) {
}).send(); }).send();
} }
void StickersListWidget::fillIcons(QList<StickerIcon> &icons) { std::vector<StickerIcon> StickersListWidget::fillIcons() {
icons.clear(); auto result = std::vector<StickerIcon>();
icons.reserve(_mySets.size() + 1); result.reserve(_mySets.size() + 1);
if (!_officialSets.empty()) { if (!_officialSets.empty()) {
icons.push_back(StickerIcon(Stickers::FeaturedSetId)); result.emplace_back(Stickers::FeaturedSetId);
} }
auto i = 0; auto i = 0;
if (i != _mySets.size() && _mySets[i].id == Stickers::FavedSetId) { if (i != _mySets.size() && _mySets[i].id == Stickers::FavedSetId) {
++i; ++i;
icons.push_back(StickerIcon(Stickers::FavedSetId)); result.emplace_back(Stickers::FavedSetId);
} }
if (i != _mySets.size() && _mySets[i].id == Stickers::RecentSetId) { if (i != _mySets.size() && _mySets[i].id == Stickers::RecentSetId) {
++i; ++i;
if (icons.empty() || icons.back().setId != Stickers::FavedSetId) { if (result.empty() || result.back().setId != Stickers::FavedSetId) {
icons.push_back(StickerIcon(Stickers::RecentSetId)); result.emplace_back(Stickers::RecentSetId);
} }
} }
for (auto l = _mySets.size(); i != l; ++i) { for (auto l = _mySets.size(); i != l; ++i) {
if (_mySets[i].id == Stickers::MegagroupSetId) { if (_mySets[i].id == Stickers::MegagroupSetId) {
icons.push_back(StickerIcon(Stickers::MegagroupSetId)); result.emplace_back(Stickers::MegagroupSetId);
icons.back().megagroup = _megagroupSet; result.back().megagroup = _megagroupSet;
continue; continue;
} }
const auto thumbnail = _mySets[i].thumbnail; const auto thumbnail = _mySets[i].thumbnail;
@ -2626,13 +2641,14 @@ void StickersListWidget::fillIcons(QList<StickerIcon> &icons) {
} }
if (pixw < 1) pixw = 1; if (pixw < 1) pixw = 1;
if (pixh < 1) pixh = 1; if (pixh < 1) pixh = 1;
icons.push_back(StickerIcon( result.emplace_back(
_mySets[i].id, _mySets[i].id,
thumbnail, thumbnail,
s, s,
pixw, pixw,
pixh)); pixh);
} }
return result;
} }
bool StickersListWidget::preventAutoHide() { bool StickersListWidget::preventAutoHide() {

View File

@ -67,7 +67,7 @@ public:
void refreshStickers(); void refreshStickers();
void fillIcons(QList<StickerIcon> &icons); std::vector<StickerIcon> fillIcons();
bool preventAutoHide(); bool preventAutoHide();
uint64 currentSet(int yOffset) const; uint64 currentSet(int yOffset) const;

@ -1 +1 @@
Subproject commit b376282b656b13c54cd892e3c2742bb26acf42fe Subproject commit dd96d4d482afae4dfd16d0d0f91ca814b9a652a5