mirror of
https://github.com/telegramdesktop/tdesktop
synced 2025-04-01 23:00:58 +00:00
Support animated stickers in inline results.
This commit is contained in:
parent
76630528f7
commit
848ea16eef
@ -524,9 +524,8 @@ void StickerSetBox::Inner::setupLottie(int index) {
|
|||||||
const auto animation = element.animated.get();
|
const auto animation = element.animated.get();
|
||||||
|
|
||||||
animation->updates(
|
animation->updates(
|
||||||
) | rpl::start_with_next_error([=](Lottie::Update update) {
|
) | rpl::start_with_next([=] {
|
||||||
this->update();
|
update();
|
||||||
}, [=](Lottie::Error error) {
|
|
||||||
}, lifetime());
|
}, lifetime());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -126,6 +126,7 @@ enum class LottieSize : uchar {
|
|||||||
StickersPanel,
|
StickersPanel,
|
||||||
StickersFooter,
|
StickersFooter,
|
||||||
SetsListThumbnail,
|
SetsListThumbnail,
|
||||||
|
InlineResults,
|
||||||
};
|
};
|
||||||
|
|
||||||
[[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
|
[[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
|
||||||
|
@ -126,7 +126,6 @@ private:
|
|||||||
int newSelected,
|
int newSelected,
|
||||||
ValidateIconAnimations animations);
|
ValidateIconAnimations animations);
|
||||||
void validateIconLottieAnimation(const StickerIcon &icon);
|
void validateIconLottieAnimation(const StickerIcon &icon);
|
||||||
QSize iconBox() const;
|
|
||||||
|
|
||||||
void refreshIconsGeometry(ValidateIconAnimations animations);
|
void refreshIconsGeometry(ValidateIconAnimations animations);
|
||||||
void refillLottieData();
|
void refillLottieData();
|
||||||
@ -701,12 +700,6 @@ void StickersListWidget::Footer::paintFeaturedStickerSetsBadge(Painter &p, int i
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QSize StickersListWidget::Footer::iconBox() const {
|
|
||||||
return QSize(
|
|
||||||
st::stickerIconWidth - 2 * st::stickerIconPadding,
|
|
||||||
st::emojiFooterHeight - 2 * st::stickerIconPadding);
|
|
||||||
}
|
|
||||||
|
|
||||||
void StickersListWidget::Footer::validateIconLottieAnimation(
|
void StickersListWidget::Footer::validateIconLottieAnimation(
|
||||||
const StickerIcon &icon) {
|
const StickerIcon &icon) {
|
||||||
if (icon.lottie
|
if (icon.lottie
|
||||||
@ -717,7 +710,10 @@ void StickersListWidget::Footer::validateIconLottieAnimation(
|
|||||||
icon.thumbnail,
|
icon.thumbnail,
|
||||||
icon.sticker,
|
icon.sticker,
|
||||||
Stickers::LottieSize::StickersFooter,
|
Stickers::LottieSize::StickersFooter,
|
||||||
iconBox() * cIntRetinaFactor(),
|
QSize(
|
||||||
|
st::stickerIconWidth - 2 * st::stickerIconPadding,
|
||||||
|
st::emojiFooterHeight - 2 * st::stickerIconPadding
|
||||||
|
) * cIntRetinaFactor(),
|
||||||
_pan->getLottieRenderer());
|
_pan->getLottieRenderer());
|
||||||
if (!player) {
|
if (!player) {
|
||||||
return;
|
return;
|
||||||
@ -768,9 +764,7 @@ void StickersListWidget::Footer::paintSetIcon(
|
|||||||
width(),
|
width(),
|
||||||
thumb->pix(origin, icon.pixw, icon.pixh));
|
thumb->pix(origin, icon.pixw, icon.pixh));
|
||||||
} else if (icon.lottie->ready()) {
|
} else if (icon.lottie->ready()) {
|
||||||
auto request = Lottie::FrameRequest();
|
const auto frame = icon.lottie->frame();
|
||||||
request.box = iconBox() * cIntRetinaFactor();
|
|
||||||
const auto frame = icon.lottie->frame(request);
|
|
||||||
const auto size = frame.size() / cIntRetinaFactor();
|
const auto size = frame.size() / cIntRetinaFactor();
|
||||||
p.drawImage(
|
p.drawImage(
|
||||||
QRect(
|
QRect(
|
||||||
|
@ -104,13 +104,12 @@ void HistorySticker::setupLottie() {
|
|||||||
_parent->data()->history()->owner().registerHeavyViewPart(_parent);
|
_parent->data()->history()->owner().registerHeavyViewPart(_parent);
|
||||||
|
|
||||||
_lottie->updates(
|
_lottie->updates(
|
||||||
) | rpl::start_with_next_error([=](Lottie::Update update) {
|
) | rpl::start_with_next([=](Lottie::Update update) {
|
||||||
update.data.match([&](const Lottie::Information &information) {
|
update.data.match([&](const Lottie::Information &information) {
|
||||||
_parent->data()->history()->owner().requestViewResize(_parent);
|
_parent->data()->history()->owner().requestViewResize(_parent);
|
||||||
}, [&](const Lottie::DisplayFrameRequest &request) {
|
}, [&](const Lottie::DisplayFrameRequest &request) {
|
||||||
_parent->data()->history()->owner().requestViewRepaint(_parent);
|
_parent->data()->history()->owner().requestViewRepaint(_parent);
|
||||||
});
|
});
|
||||||
}, [=](Lottie::Error error) {
|
|
||||||
}, _lifetime);
|
}, _lifetime);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -126,7 +125,7 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, c
|
|||||||
auto sticker = _data->sticker();
|
auto sticker = _data->sticker();
|
||||||
if (!sticker) return;
|
if (!sticker) return;
|
||||||
|
|
||||||
if (sticker->animated && _data->loaded() && !_lottie) {
|
if (sticker->animated && !_lottie && _data->loaded()) {
|
||||||
const_cast<HistorySticker*>(this)->setupLottie();
|
const_cast<HistorySticker*>(this)->setupLottie();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,6 +15,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||||||
#include "styles/style_chat_helpers.h"
|
#include "styles/style_chat_helpers.h"
|
||||||
#include "styles/style_widgets.h"
|
#include "styles/style_widgets.h"
|
||||||
#include "inline_bots/inline_bot_result.h"
|
#include "inline_bots/inline_bot_result.h"
|
||||||
|
#include "lottie/lottie_single_player.h"
|
||||||
#include "media/audio/media_audio.h"
|
#include "media/audio/media_audio.h"
|
||||||
#include "media/clip/media_clip_reader.h"
|
#include "media/clip/media_clip_reader.h"
|
||||||
#include "media/player/media_player_instance.h"
|
#include "media/player/media_player_instance.h"
|
||||||
@ -387,6 +388,8 @@ Sticker::Sticker(not_null<Context*> context, Result *result)
|
|||||||
: FileBase(context, result) {
|
: FileBase(context, result) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Sticker::~Sticker() = default;
|
||||||
|
|
||||||
void Sticker::initDimensions() {
|
void Sticker::initDimensions() {
|
||||||
_maxw = st::stickerPanSize.width();
|
_maxw = st::stickerPanSize.width();
|
||||||
_minh = st::stickerPanSize.height();
|
_minh = st::stickerPanSize.height();
|
||||||
@ -411,7 +414,17 @@ void Sticker::paint(Painter &p, const QRect &clip, const PaintContext *context)
|
|||||||
}
|
}
|
||||||
|
|
||||||
prepareThumbnail();
|
prepareThumbnail();
|
||||||
if (!_thumb.isNull()) {
|
if (_lottie && _lottie->ready()) {
|
||||||
|
const auto frame = _lottie->frame();
|
||||||
|
_lottie->markFrameShown();
|
||||||
|
const auto size = frame.size() / cIntRetinaFactor();
|
||||||
|
const auto pos = QPoint(
|
||||||
|
(st::stickerPanSize.width() - size.width()) / 2,
|
||||||
|
(st::stickerPanSize.height() - size.height()) / 2);
|
||||||
|
p.drawImage(
|
||||||
|
QRect(pos, size),
|
||||||
|
frame);
|
||||||
|
} else if (!_thumb.isNull()) {
|
||||||
int w = _thumb.width() / cIntRetinaFactor(), h = _thumb.height() / cIntRetinaFactor();
|
int w = _thumb.width() / cIntRetinaFactor(), h = _thumb.height() / cIntRetinaFactor();
|
||||||
QPoint pos = QPoint((st::stickerPanSize.width() - w) / 2, (st::stickerPanSize.height() - h) / 2);
|
QPoint pos = QPoint((st::stickerPanSize.width() - w) / 2, (st::stickerPanSize.height() - h) / 2);
|
||||||
p.drawPixmap(pos, _thumb);
|
p.drawPixmap(pos, _thumb);
|
||||||
@ -450,11 +463,31 @@ QSize Sticker::getThumbSize() const {
|
|||||||
return QSize(qMax(w, 1), qMax(h, 1));
|
return QSize(qMax(w, 1), qMax(h, 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Sticker::setupLottie(not_null<DocumentData*> document) const {
|
||||||
|
_lottie = Stickers::LottiePlayerFromDocument(
|
||||||
|
document,
|
||||||
|
Stickers::LottieSize::InlineResults,
|
||||||
|
QSize(
|
||||||
|
st::stickerPanSize.width() - st::buttonRadius * 2,
|
||||||
|
st::stickerPanSize.height() - st::buttonRadius * 2
|
||||||
|
) * cIntRetinaFactor());
|
||||||
|
|
||||||
|
_lottie->updates(
|
||||||
|
) | rpl::start_with_next([=] {
|
||||||
|
update();
|
||||||
|
}, _lifetime);
|
||||||
|
}
|
||||||
|
|
||||||
void Sticker::prepareThumbnail() const {
|
void Sticker::prepareThumbnail() const {
|
||||||
if (const auto document = getShownDocument()) {
|
if (const auto document = getShownDocument()) {
|
||||||
|
if (document->sticker()->animated
|
||||||
|
&& !_lottie
|
||||||
|
&& document->loaded()) {
|
||||||
|
setupLottie(document);
|
||||||
|
}
|
||||||
document->checkStickerSmall();
|
document->checkStickerSmall();
|
||||||
if (const auto sticker = document->getStickerSmall()) {
|
if (const auto sticker = document->getStickerSmall()) {
|
||||||
if (!_thumbLoaded && sticker->loaded()) {
|
if (!_lottie && !_thumbLoaded && sticker->loaded()) {
|
||||||
const auto thumbSize = getThumbSize();
|
const auto thumbSize = getThumbSize();
|
||||||
_thumb = sticker->pix(
|
_thumb = sticker->pix(
|
||||||
document->stickerSetOrigin(),
|
document->stickerSetOrigin(),
|
||||||
|
@ -13,6 +13,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||||||
#include "ui/effects/radial_animation.h"
|
#include "ui/effects/radial_animation.h"
|
||||||
#include "ui/text/text.h"
|
#include "ui/text/text.h"
|
||||||
|
|
||||||
|
namespace Lottie {
|
||||||
|
class SinglePlayer;
|
||||||
|
} // namespace Lottie
|
||||||
|
|
||||||
namespace InlineBots {
|
namespace InlineBots {
|
||||||
namespace Layout {
|
namespace Layout {
|
||||||
namespace internal {
|
namespace internal {
|
||||||
@ -151,6 +155,7 @@ private:
|
|||||||
class Sticker : public FileBase {
|
class Sticker : public FileBase {
|
||||||
public:
|
public:
|
||||||
Sticker(not_null<Context*> context, Result *result);
|
Sticker(not_null<Context*> context, Result *result);
|
||||||
|
~Sticker();
|
||||||
// Not used anywhere currently.
|
// Not used anywhere currently.
|
||||||
//Sticker(not_null<Context*> context, DocumentData *document);
|
//Sticker(not_null<Context*> context, DocumentData *document);
|
||||||
|
|
||||||
@ -173,14 +178,18 @@ public:
|
|||||||
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
void setupLottie(not_null<DocumentData*> document) const;
|
||||||
QSize getThumbSize() const;
|
QSize getThumbSize() const;
|
||||||
|
void prepareThumbnail() const;
|
||||||
|
|
||||||
mutable Ui::Animations::Simple _a_over;
|
mutable Ui::Animations::Simple _a_over;
|
||||||
mutable bool _active = false;
|
mutable bool _active = false;
|
||||||
|
|
||||||
mutable QPixmap _thumb;
|
mutable QPixmap _thumb;
|
||||||
mutable bool _thumbLoaded = false;
|
mutable bool _thumbLoaded = false;
|
||||||
void prepareThumbnail() const;
|
|
||||||
|
mutable std::unique_ptr<Lottie::SinglePlayer> _lottie;
|
||||||
|
mutable rpl::lifetime _lifetime;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -879,10 +879,8 @@ void MediaPreviewWidget::paintEvent(QPaintEvent *e) {
|
|||||||
if (!_lottie || !_lottie->ready()) {
|
if (!_lottie || !_lottie->ready()) {
|
||||||
return QImage();
|
return QImage();
|
||||||
}
|
}
|
||||||
auto request = Lottie::FrameRequest();
|
|
||||||
request.box = currentDimensions() * cIntRetinaFactor();
|
|
||||||
_lottie->markFrameShown();
|
_lottie->markFrameShown();
|
||||||
return _lottie->frame(request);
|
return _lottie->frame();
|
||||||
}();
|
}();
|
||||||
const auto pixmap = image.isNull() ? currentImage() : QPixmap();
|
const auto pixmap = image.isNull() ? currentImage() : QPixmap();
|
||||||
const auto size = image.isNull() ? pixmap.size() : image.size();
|
const auto size = image.isNull() ? pixmap.size() : image.size();
|
||||||
@ -1054,13 +1052,12 @@ void MediaPreviewWidget::setupLottie() {
|
|||||||
Lottie::FrameRequest{ currentDimensions() * cIntRetinaFactor() });
|
Lottie::FrameRequest{ currentDimensions() * cIntRetinaFactor() });
|
||||||
|
|
||||||
_lottie->updates(
|
_lottie->updates(
|
||||||
) | rpl::start_with_next_error([=](Lottie::Update update) {
|
) | rpl::start_with_next([=](Lottie::Update update) {
|
||||||
update.data.match([&](const Lottie::Information &) {
|
update.data.match([&](const Lottie::Information &) {
|
||||||
this->update();
|
this->update();
|
||||||
}, [&](const Lottie::DisplayFrameRequest &) {
|
}, [&](const Lottie::DisplayFrameRequest &) {
|
||||||
this->update(updateArea());
|
this->update(updateArea());
|
||||||
});
|
});
|
||||||
}, [=](Lottie::Error error) {
|
|
||||||
}, lifetime());
|
}, lifetime());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user