mirror of https://github.com/procxx/kepka.git
Show animated sticker preview.
This commit is contained in:
parent
8f3f898c47
commit
040cae6a9a
|
@ -19,6 +19,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "ui/image/image.h"
|
#include "ui/image/image.h"
|
||||||
#include "ui/emoji_config.h"
|
#include "ui/emoji_config.h"
|
||||||
#include "window/window_main_menu.h"
|
#include "window/window_main_menu.h"
|
||||||
|
#include "lottie/lottie_animation.h"
|
||||||
#include "auth_session.h"
|
#include "auth_session.h"
|
||||||
#include "chat_helpers/stickers.h"
|
#include "chat_helpers/stickers.h"
|
||||||
#include "window/window_session_controller.h"
|
#include "window/window_session_controller.h"
|
||||||
|
@ -856,7 +857,7 @@ LayerStackWidget::~LayerStackWidget() {
|
||||||
MediaPreviewWidget::MediaPreviewWidget(
|
MediaPreviewWidget::MediaPreviewWidget(
|
||||||
QWidget *parent,
|
QWidget *parent,
|
||||||
not_null<Window::SessionController*> controller)
|
not_null<Window::SessionController*> controller)
|
||||||
: TWidget(parent)
|
: RpWidget(parent)
|
||||||
, _controller(controller)
|
, _controller(controller)
|
||||||
, _emojiSize(Ui::Emoji::GetSizeLarge() / cIntRetinaFactor()) {
|
, _emojiSize(Ui::Emoji::GetSizeLarge() / cIntRetinaFactor()) {
|
||||||
setAttribute(Qt::WA_TransparentForMouseEvents);
|
setAttribute(Qt::WA_TransparentForMouseEvents);
|
||||||
|
@ -867,8 +868,18 @@ void MediaPreviewWidget::paintEvent(QPaintEvent *e) {
|
||||||
Painter p(this);
|
Painter p(this);
|
||||||
QRect r(e->rect());
|
QRect r(e->rect());
|
||||||
|
|
||||||
auto image = currentImage();
|
const auto image = [&] {
|
||||||
int w = image.width() / cIntRetinaFactor(), h = image.height() / cIntRetinaFactor();
|
if (!_lottie || !_lottie->ready()) {
|
||||||
|
return QImage();
|
||||||
|
}
|
||||||
|
auto request = Lottie::FrameRequest();
|
||||||
|
request.resize = currentDimensions() * cIntRetinaFactor();
|
||||||
|
_lottie->markFrameShown();
|
||||||
|
return _lottie->frame(request);
|
||||||
|
}();
|
||||||
|
const auto pixmap = image.isNull() ? currentImage() : QPixmap();
|
||||||
|
const auto size = image.isNull() ? pixmap.size() : image.size();
|
||||||
|
int w = size.width() / cIntRetinaFactor(), h = size.height() / cIntRetinaFactor();
|
||||||
auto shown = _a_shown.value(_hiding ? 0. : 1.);
|
auto shown = _a_shown.value(_hiding ? 0. : 1.);
|
||||||
if (!_a_shown.animating()) {
|
if (!_a_shown.animating()) {
|
||||||
if (_hiding) {
|
if (_hiding) {
|
||||||
|
@ -882,7 +893,13 @@ void MediaPreviewWidget::paintEvent(QPaintEvent *e) {
|
||||||
// h = qMax(qRound(h * (st::stickerPreviewMin + ((1. - st::stickerPreviewMin) * shown)) / 2.) * 2 + int(h % 2), 1);
|
// h = qMax(qRound(h * (st::stickerPreviewMin + ((1. - st::stickerPreviewMin) * shown)) / 2.) * 2 + int(h % 2), 1);
|
||||||
}
|
}
|
||||||
p.fillRect(r, st::stickerPreviewBg);
|
p.fillRect(r, st::stickerPreviewBg);
|
||||||
p.drawPixmap((width() - w) / 2, (height() - h) / 2, image);
|
if (image.isNull()) {
|
||||||
|
p.drawPixmap((width() - w) / 2, (height() - h) / 2, pixmap);
|
||||||
|
} else {
|
||||||
|
p.drawImage(
|
||||||
|
QRect((width() - w) / 2, (height() - h) / 2, w, h),
|
||||||
|
image);
|
||||||
|
}
|
||||||
if (!_emojiList.empty()) {
|
if (!_emojiList.empty()) {
|
||||||
const auto emojiCount = _emojiList.size();
|
const auto emojiCount = _emojiList.size();
|
||||||
const auto emojiWidth = (emojiCount * _emojiSize) + (emojiCount - 1) * st::stickerEmojiSkip;
|
const auto emojiWidth = (emojiCount * _emojiSize) + (emojiCount - 1) * st::stickerEmojiSkip;
|
||||||
|
@ -977,6 +994,7 @@ void MediaPreviewWidget::fillEmojiString() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void MediaPreviewWidget::resetGifAndCache() {
|
void MediaPreviewWidget::resetGifAndCache() {
|
||||||
|
_lottie = nullptr;
|
||||||
_gif.reset();
|
_gif.reset();
|
||||||
_cacheStatus = CacheNotLoaded;
|
_cacheStatus = CacheNotLoaded;
|
||||||
_cachedSize = QSize();
|
_cachedSize = QSize();
|
||||||
|
@ -1021,11 +1039,30 @@ QSize MediaPreviewWidget::currentDimensions() const {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MediaPreviewWidget::setupLottie() {
|
||||||
|
Expects(_document != nullptr);
|
||||||
|
|
||||||
|
_lottie = _document->data().isEmpty()
|
||||||
|
? Lottie::FromFile(_document->filepath())
|
||||||
|
: Lottie::FromData(_document->data());
|
||||||
|
|
||||||
|
_lottie->updates(
|
||||||
|
) | rpl::start_with_next_error([=](Lottie::Update update) {
|
||||||
|
this->update();
|
||||||
|
}, [=](Lottie::Error error) {
|
||||||
|
}, lifetime());
|
||||||
|
}
|
||||||
|
|
||||||
QPixmap MediaPreviewWidget::currentImage() const {
|
QPixmap MediaPreviewWidget::currentImage() const {
|
||||||
if (_document) {
|
if (_document) {
|
||||||
if (_document->sticker()) {
|
if (const auto sticker = _document->sticker()) {
|
||||||
if (_cacheStatus != CacheLoaded) {
|
if (_cacheStatus != CacheLoaded) {
|
||||||
if (const auto image = _document->getStickerLarge()) {
|
if (sticker->animated && !_lottie && _document->loaded()) {
|
||||||
|
const_cast<MediaPreviewWidget*>(this)->setupLottie();
|
||||||
|
}
|
||||||
|
if (_lottie && _lottie->ready()) {
|
||||||
|
return QPixmap();
|
||||||
|
} else if (const auto image = _document->getStickerLarge()) {
|
||||||
QSize s = currentDimensions();
|
QSize s = currentDimensions();
|
||||||
_cache = image->pix(_origin, s.width(), s.height());
|
_cache = image->pix(_origin, s.width(), s.height());
|
||||||
_cacheStatus = CacheLoaded;
|
_cacheStatus = CacheLoaded;
|
||||||
|
|
|
@ -11,6 +11,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "ui/effects/animations.h"
|
#include "ui/effects/animations.h"
|
||||||
#include "data/data_file_origin.h"
|
#include "data/data_file_origin.h"
|
||||||
|
|
||||||
|
namespace Lottie {
|
||||||
|
class Animation;
|
||||||
|
} // namespace Lottie
|
||||||
|
|
||||||
namespace Window {
|
namespace Window {
|
||||||
|
|
||||||
class MainMenu;
|
class MainMenu;
|
||||||
|
@ -195,9 +199,11 @@ private:
|
||||||
|
|
||||||
} // namespace Window
|
} // namespace Window
|
||||||
|
|
||||||
class MediaPreviewWidget : public TWidget, private base::Subscriber {
|
class MediaPreviewWidget : public Ui::RpWidget, private base::Subscriber {
|
||||||
public:
|
public:
|
||||||
MediaPreviewWidget(QWidget *parent, not_null<Window::SessionController*> controller);
|
MediaPreviewWidget(
|
||||||
|
QWidget *parent,
|
||||||
|
not_null<Window::SessionController*> controller);
|
||||||
|
|
||||||
void showPreview(
|
void showPreview(
|
||||||
Data::FileOrigin origin,
|
Data::FileOrigin origin,
|
||||||
|
@ -216,6 +222,7 @@ protected:
|
||||||
private:
|
private:
|
||||||
QSize currentDimensions() const;
|
QSize currentDimensions() const;
|
||||||
QPixmap currentImage() const;
|
QPixmap currentImage() const;
|
||||||
|
void setupLottie();
|
||||||
void startShow();
|
void startShow();
|
||||||
void fillEmojiString();
|
void fillEmojiString();
|
||||||
void resetGifAndCache();
|
void resetGifAndCache();
|
||||||
|
@ -228,6 +235,7 @@ private:
|
||||||
DocumentData *_document = nullptr;
|
DocumentData *_document = nullptr;
|
||||||
PhotoData *_photo = nullptr;
|
PhotoData *_photo = nullptr;
|
||||||
Media::Clip::ReaderPointer _gif;
|
Media::Clip::ReaderPointer _gif;
|
||||||
|
std::unique_ptr<Lottie::Animation> _lottie;
|
||||||
|
|
||||||
int _emojiSize;
|
int _emojiSize;
|
||||||
std::vector<not_null<EmojiPtr>> _emojiList;
|
std::vector<not_null<EmojiPtr>> _emojiList;
|
||||||
|
|
Loading…
Reference in New Issue