1556 lines
		
	
	
	
		
			43 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1556 lines
		
	
	
	
		
			43 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #include "boxes/sticker_set_box.h"
 | |
| 
 | |
| #include "kotato/kotato_lang.h"
 | |
| #include "data/data_document.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_file_origin.h"
 | |
| #include "data/data_document_media.h"
 | |
| #include "data/data_peer_values.h"
 | |
| #include "data/stickers/data_stickers.h"
 | |
| #include "data/stickers/data_custom_emoji.h"
 | |
| #include "menu/menu_send.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "ui/boxes/confirm_box.h"
 | |
| #include "boxes/premium_preview_box.h"
 | |
| #include "core/application.h"
 | |
| #include "mtproto/sender.h"
 | |
| #include "storage/storage_account.h"
 | |
| #include "dialogs/ui/dialogs_layout.h"
 | |
| #include "ui/widgets/buttons.h"
 | |
| #include "ui/widgets/scroll_area.h"
 | |
| #include "ui/widgets/gradient_round_button.h"
 | |
| #include "ui/widgets/dropdown_menu.h"
 | |
| #include "ui/image/image.h"
 | |
| #include "ui/image/image_location_factory.h"
 | |
| #include "ui/text/text_utilities.h"
 | |
| #include "ui/text/custom_emoji_instance.h"
 | |
| #include "ui/effects/path_shift_gradient.h"
 | |
| #include "ui/emoji_config.h"
 | |
| #include "ui/painter.h"
 | |
| #include "ui/power_saving.h"
 | |
| #include "ui/toast/toast.h"
 | |
| #include "ui/widgets/popup_menu.h"
 | |
| #include "ui/cached_round_corners.h"
 | |
| #include "lottie/lottie_multi_player.h"
 | |
| #include "lottie/lottie_animation.h"
 | |
| #include "chat_helpers/compose/compose_show.h"
 | |
| #include "chat_helpers/stickers_lottie.h"
 | |
| #include "chat_helpers/stickers_list_widget.h"
 | |
| #include "media/clip/media_clip_reader.h"
 | |
| #include "window/window_controller.h"
 | |
| #include "settings/settings_premium.h"
 | |
| #include "base/unixtime.h"
 | |
| #include "main/main_session.h"
 | |
| #include "apiwrap.h"
 | |
| #include "api/api_toggling_media.h"
 | |
| #include "api/api_common.h"
 | |
| #include "mainwidget.h"
 | |
| #include "mainwindow.h"
 | |
| #include "styles/style_info.h"
 | |
| #include "styles/style_layers.h"
 | |
| #include "styles/style_chat_helpers.h"
 | |
| #include "styles/style_info.h"
 | |
| #include "styles/style_menu_icons.h"
 | |
| #include "styles/style_premium.h"
 | |
| 
 | |
| #include <QtWidgets/QApplication>
 | |
| #include <QtGui/QClipboard>
 | |
| #include <QtSvg/QSvgRenderer>
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kStickersPerRow = 5;
 | |
| constexpr auto kEmojiPerRow = 8;
 | |
| constexpr auto kMinRepaintDelay = crl::time(33);
 | |
| constexpr auto kMinAfterScrollDelay = crl::time(33);
 | |
| constexpr auto kGrayLockOpacity = 0.3;
 | |
| 
 | |
| using Data::StickersSet;
 | |
| using Data::StickersPack;
 | |
| using SetFlag = Data::StickersSetFlag;
 | |
| 
 | |
| [[nodiscard]] std::optional<QColor> ComputeImageColor(const QImage &frame) {
 | |
| 	if (frame.isNull()
 | |
| 		|| frame.format() != QImage::Format_ARGB32_Premultiplied) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	auto sr = int64();
 | |
| 	auto sg = int64();
 | |
| 	auto sb = int64();
 | |
| 	auto sa = int64();
 | |
| 	const auto factor = frame.devicePixelRatio();
 | |
| 	const auto size = st::stickersPremiumLock.size() * factor;
 | |
| 	const auto width = std::min(frame.width(), size.width());
 | |
| 	const auto height = std::min(frame.height(), size.height());
 | |
| 	const auto skipx = (frame.width() - width) / 2;
 | |
| 	const auto radius = st::roundRadiusSmall;
 | |
| 	const auto skipy = std::max(frame.height() - height - radius, 0);
 | |
| 	const auto perline = frame.bytesPerLine();
 | |
| 	const auto addperline = perline - (width * 4);
 | |
| 	auto bits = static_cast<const uchar*>(frame.bits())
 | |
| 		+ perline * skipy
 | |
| 		+ sizeof(uint32) * skipx;
 | |
| 	for (auto y = 0; y != height; ++y) {
 | |
| 		for (auto x = 0; x != width; ++x) {
 | |
| 			sb += int(*bits++);
 | |
| 			sg += int(*bits++);
 | |
| 			sr += int(*bits++);
 | |
| 			sa += int(*bits++);
 | |
| 		}
 | |
| 		bits += addperline;
 | |
| 	}
 | |
| 	if (!sa) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	return QColor(sr * 255 / sa, sg * 255 / sa, sb * 255 / sa, 255);
 | |
| 
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QColor ComputeLockColor(const QImage &frame) {
 | |
| 	return ComputeImageColor(frame).value_or(st::windowSubTextFg->c);
 | |
| }
 | |
| 
 | |
| void ValidatePremiumLockBg(QImage &image, const QImage &frame) {
 | |
| 	if (!image.isNull()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto factor = style::DevicePixelRatio();
 | |
| 	const auto size = st::stickersPremiumLock.size();
 | |
| 	image = QImage(
 | |
| 		size * factor,
 | |
| 		QImage::Format_ARGB32_Premultiplied);
 | |
| 	image.setDevicePixelRatio(factor);
 | |
| 	auto p = QPainter(&image);
 | |
| 	const auto color = ComputeLockColor(frame);
 | |
| 	p.fillRect(
 | |
| 		QRect(QPoint(), size),
 | |
| 		anim::color(color, st::windowSubTextFg, kGrayLockOpacity));
 | |
| 	p.end();
 | |
| 
 | |
| 	image = Images::Circle(std::move(image));
 | |
| }
 | |
| 
 | |
| void ValidatePremiumStarFg(QImage &image) {
 | |
| 	if (!image.isNull()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto factor = style::DevicePixelRatio();
 | |
| 	const auto size = st::stickersPremiumLock.size();
 | |
| 	image = QImage(
 | |
| 		size * factor,
 | |
| 		QImage::Format_ARGB32_Premultiplied);
 | |
| 	image.setDevicePixelRatio(factor);
 | |
| 	image.fill(Qt::transparent);
 | |
| 	auto p = QPainter(&image);
 | |
| 	auto star = QSvgRenderer(u":/gui/icons/settings/star.svg"_q);
 | |
| 	const auto skip = size.width() / 5.;
 | |
| 	const auto outer = QRectF(QPointF(), size).marginsRemoved(
 | |
| 		{ skip, skip, skip, skip });
 | |
| 	p.setBrush(st::premiumButtonFg);
 | |
| 	p.setPen(Qt::NoPen);
 | |
| 	star.render(&p, outer);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] TextForMimeData PrepareTextFromEmoji(
 | |
| 		not_null<DocumentData*> document) {
 | |
| 	const auto info = document->sticker();
 | |
| 	const auto text = info ? info->alt : QString();
 | |
| 	return {
 | |
| 		.expanded = text,
 | |
| 		.rich = {
 | |
| 			text,
 | |
| 			{
 | |
| 				EntityInText(
 | |
| 					EntityType::CustomEmoji,
 | |
| 					0,
 | |
| 					text.size(),
 | |
| 					Data::SerializeCustomEmojiId(document))
 | |
| 			},
 | |
| 		},
 | |
| 	};
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| StickerPremiumMark::StickerPremiumMark(not_null<Main::Session*> session) {
 | |
| 	style::PaletteChanged(
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		_lockGray = QImage();
 | |
| 		_star = QImage();
 | |
| 	}, _lifetime);
 | |
| 
 | |
| 	Data::AmPremiumValue(
 | |
| 		session
 | |
| 	) | rpl::start_with_next([=](bool premium) {
 | |
| 		_premium = premium;
 | |
| 	}, _lifetime);
 | |
| }
 | |
| 
 | |
| void StickerPremiumMark::paint(
 | |
| 		QPainter &p,
 | |
| 		const QImage &frame,
 | |
| 		QImage &backCache,
 | |
| 		QPoint position,
 | |
| 		QSize singleSize,
 | |
| 		int outerWidth) {
 | |
| 	validateLock(frame, backCache);
 | |
| 	const auto &bg = frame.isNull() ? _lockGray : backCache;
 | |
| 	const auto factor = style::DevicePixelRatio();
 | |
| 	const auto radius = st::roundRadiusSmall;
 | |
| 	const auto point = position + QPoint(
 | |
| 		(_premium
 | |
| 			? (singleSize.width() - (bg.width() / factor) - radius)
 | |
| 			: (singleSize.width() - (bg.width() / factor)) / 2),
 | |
| 		singleSize.height() - (bg.height() / factor) - radius);
 | |
| 	p.drawImage(point, bg);
 | |
| 	if (_premium) {
 | |
| 		validateStar();
 | |
| 		p.drawImage(point, _star);
 | |
| 	} else {
 | |
| 		st::stickersPremiumLock.paint(p, point, outerWidth);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerPremiumMark::validateLock(
 | |
| 		const QImage &frame,
 | |
| 		QImage &backCache) {
 | |
| 	auto &image = frame.isNull() ? _lockGray : backCache;
 | |
| 	ValidatePremiumLockBg(image, frame);
 | |
| }
 | |
| 
 | |
| void StickerPremiumMark::validateStar() {
 | |
| 	ValidatePremiumStarFg(_star);
 | |
| }
 | |
| 
 | |
| class StickerSetBox::Inner final : public Ui::RpWidget {
 | |
| public:
 | |
| 	Inner(
 | |
| 		QWidget *parent,
 | |
| 		std::shared_ptr<ChatHelpers::Show> show,
 | |
| 		const StickerSetIdentifier &set,
 | |
| 		Data::StickersType type);
 | |
| 
 | |
| 	[[nodiscard]] bool loaded() const;
 | |
| 	[[nodiscard]] bool notInstalled() const;
 | |
| 	[[nodiscard]] bool premiumEmojiSet() const;
 | |
| 	[[nodiscard]] bool official() const;
 | |
| 	[[nodiscard]] rpl::producer<TextWithEntities> title() const;
 | |
| 	[[nodiscard]] QString shortName() const;
 | |
| 	[[nodiscard]] bool isEmojiSet() const;
 | |
| 	[[nodiscard]] uint64 setId() const;
 | |
| 
 | |
| 	void install();
 | |
| 	[[nodiscard]] rpl::producer<uint64> setInstalled() const;
 | |
| 	[[nodiscard]] rpl::producer<uint64> setArchived() const;
 | |
| 	[[nodiscard]] rpl::producer<> updateControls() const;
 | |
| 
 | |
| 	[[nodiscard]] rpl::producer<Error> errors() const;
 | |
| 
 | |
| 	void archiveStickers();
 | |
| 
 | |
| 	[[nodiscard]] Data::StickersType setType() const {
 | |
| 		return (_setFlags & SetFlag::Emoji)
 | |
| 			? Data::StickersType::Emoji
 | |
| 			: (_setFlags & SetFlag::Masks)
 | |
| 			? Data::StickersType::Masks
 | |
| 			: Data::StickersType::Stickers;
 | |
| 	}
 | |
| 
 | |
| 	~Inner();
 | |
| 
 | |
| protected:
 | |
| 	void mousePressEvent(QMouseEvent *e) override;
 | |
| 	void mouseMoveEvent(QMouseEvent *e) override;
 | |
| 	void mouseReleaseEvent(QMouseEvent *e) override;
 | |
| 	void contextMenuEvent(QContextMenuEvent *e) override;
 | |
| 	void paintEvent(QPaintEvent *e) override;
 | |
| 	void leaveEventHook(QEvent *e) override;
 | |
| 
 | |
| private:
 | |
| 	struct Element {
 | |
| 		not_null<DocumentData*> document;
 | |
| 		std::shared_ptr<Data::DocumentMedia> documentMedia;
 | |
| 		Lottie::Animation *lottie = nullptr;
 | |
| 		Media::Clip::ReaderPointer webm;
 | |
| 		Ui::Text::CustomEmoji *emoji = nullptr;
 | |
| 		Ui::Animations::Simple overAnimation;
 | |
| 
 | |
| 		mutable QImage premiumLock;
 | |
| 	};
 | |
| 
 | |
| 	void visibleTopBottomUpdated(int visibleTop, int visibleBottom) override;
 | |
| 
 | |
| 	[[nodiscard]] Ui::MessageSendingAnimationFrom messageSentAnimationInfo(
 | |
| 		int index,
 | |
| 		not_null<DocumentData*> document) const;
 | |
| 	[[nodiscard]] QSize boundingBoxSize() const;
 | |
| 
 | |
| 	void paintSticker(
 | |
| 		Painter &p,
 | |
| 		int index,
 | |
| 		QPoint position,
 | |
| 		bool paused,
 | |
| 		crl::time now) const;
 | |
| 	void setupLottie(int index);
 | |
| 	void setupWebm(int index);
 | |
| 	void clipCallback(
 | |
| 		Media::Clip::Notification notification,
 | |
| 		not_null<DocumentData*> document,
 | |
| 		int index);
 | |
| 	void setupEmoji(int index);
 | |
| 	[[nodiscard]] not_null<Ui::Text::CustomEmoji*> resolveCustomEmoji(
 | |
| 		not_null<DocumentData*> document);
 | |
| 	void customEmojiRepaint();
 | |
| 
 | |
| 	void updateSelected();
 | |
| 	void setSelected(int selected);
 | |
| 	void startOverAnimation(int index, float64 from, float64 to);
 | |
| 	int stickerFromGlobalPos(const QPoint &p) const;
 | |
| 
 | |
| 	void gotSet(const MTPmessages_StickerSet &set);
 | |
| 	void installDone(const MTPmessages_StickerSetInstallResult &result);
 | |
| 
 | |
| 	void chosen(
 | |
| 		int index,
 | |
| 		not_null<DocumentData*> sticker,
 | |
| 		Api::SendOptions options);
 | |
| 
 | |
| 	not_null<Lottie::MultiPlayer*> getLottiePlayer();
 | |
| 
 | |
| 	void showPreview();
 | |
| 	void showPreviewAt(QPoint globalPos);
 | |
| 
 | |
| 	void updateItems();
 | |
| 	void repaintItems(crl::time now = 0);
 | |
| 
 | |
| 	const std::shared_ptr<ChatHelpers::Show> _show;
 | |
| 	const not_null<Main::Session*> _session;
 | |
| 
 | |
| 	MTP::Sender _api;
 | |
| 	std::vector<Element> _elements;
 | |
| 	std::unique_ptr<Lottie::MultiPlayer> _lottiePlayer;
 | |
| 
 | |
| 	base::flat_map<
 | |
| 		not_null<DocumentData*>,
 | |
| 		std::unique_ptr<Ui::Text::CustomEmoji>> _customEmoji;
 | |
| 	bool _repaintScheduled = false;
 | |
| 
 | |
| 	StickersPack _pack;
 | |
| 	base::flat_map<EmojiPtr, StickersPack> _emoji;
 | |
| 	bool _loaded = false;
 | |
| 	uint64 _setId = 0;
 | |
| 	uint64 _setAccessHash = 0;
 | |
| 	uint64 _setHash = 0;
 | |
| 	DocumentId _setThumbnailDocumentId = 0;
 | |
| 	QString _setTitle, _setShortName;
 | |
| 	int _setCount = 0;
 | |
| 	Data::StickersSetFlags _setFlags;
 | |
| 	int _rowsCount = 0;
 | |
| 	int _perRow = 0;
 | |
| 	QSize _singleSize;
 | |
| 	TimeId _setInstallDate = TimeId(0);
 | |
| 	StickerType _setThumbnailType = StickerType::Webp;
 | |
| 	ImageWithLocation _setThumbnail;
 | |
| 
 | |
| 	const std::unique_ptr<Ui::PathShiftGradient> _pathGradient;
 | |
| 	mutable StickerPremiumMark _premiumMark;
 | |
| 
 | |
| 	int _visibleTop = 0;
 | |
| 	int _visibleBottom = 0;
 | |
| 	crl::time _lastScrolledAt = 0;
 | |
| 	crl::time _lastUpdatedAt = 0;
 | |
| 	base::Timer _updateItemsTimer;
 | |
| 
 | |
| 	StickerSetIdentifier _input;
 | |
| 	QMargins _padding;
 | |
| 
 | |
| 	mtpRequestId _installRequest = 0;
 | |
| 
 | |
| 	int _selected = -1;
 | |
| 
 | |
| 	base::Timer _previewTimer;
 | |
| 	int _previewShown = -1;
 | |
| 
 | |
| 	base::unique_qptr<Ui::PopupMenu> _menu;
 | |
| 
 | |
| 	rpl::event_stream<uint64> _setInstalled;
 | |
| 	rpl::event_stream<uint64> _setArchived;
 | |
| 	rpl::event_stream<> _updateControls;
 | |
| 	rpl::event_stream<Error> _errors;
 | |
| 
 | |
| };
 | |
| 
 | |
| StickerSetBox::StickerSetBox(
 | |
| 	QWidget *parent,
 | |
| 	std::shared_ptr<ChatHelpers::Show> show,
 | |
| 	const StickerSetIdentifier &set,
 | |
| 	Data::StickersType type)
 | |
| : _show(std::move(show))
 | |
| , _session(&_show->session())
 | |
| , _set(set)
 | |
| , _type(type) {
 | |
| }
 | |
| 
 | |
| StickerSetBox::StickerSetBox(
 | |
| 	QWidget *parent,
 | |
| 	std::shared_ptr<ChatHelpers::Show> show,
 | |
| 	not_null<Data::StickersSet*> set)
 | |
| : StickerSetBox(parent, std::move(show), set->identifier(), set->type()) {
 | |
| }
 | |
| 
 | |
| QPointer<Ui::BoxContent> StickerSetBox::Show(
 | |
| 		std::shared_ptr<ChatHelpers::Show> show,
 | |
| 		not_null<DocumentData*> document) {
 | |
| 	if (const auto sticker = document->sticker()) {
 | |
| 		if (sticker->set) {
 | |
| 			auto box = Box<StickerSetBox>(
 | |
| 				show,
 | |
| 				sticker->set,
 | |
| 				sticker->setType);
 | |
| 			const auto result = QPointer<Ui::BoxContent>(box.data());
 | |
| 			show->showBox(std::move(box));
 | |
| 			return result;
 | |
| 		}
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| void StickerSetBox::prepare() {
 | |
| 	setTitle(tr::lng_contacts_loading());
 | |
| 
 | |
| 	_inner = setInnerWidget(
 | |
| 		object_ptr<Inner>(this, _show, _set, _type),
 | |
| 		st::stickersScroll);
 | |
| 	_session->data().stickers().updated(
 | |
| 		_type
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		updateButtons();
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	setDimensions(
 | |
| 		st::boxWideWidth,
 | |
| 		(_type == Data::StickersType::Emoji
 | |
| 			? st::emojiSetMaxHeight
 | |
| 			: st::stickersMaxHeight));
 | |
| 
 | |
| 	updateTitleAndButtons();
 | |
| 
 | |
| 	_inner->updateControls(
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		updateTitleAndButtons();
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	_inner->setInstalled(
 | |
| 	) | rpl::start_with_next([=](uint64 setId) {
 | |
| 		if (_inner->setType() == Data::StickersType::Masks) {
 | |
| 			showToast(tr::lng_masks_installed(tr::now));
 | |
| 		} else if (_inner->setType() == Data::StickersType::Emoji) {
 | |
| 			auto &stickers = _session->data().stickers();
 | |
| 			stickers.notifyEmojiSetInstalled(setId);
 | |
| 		} else if (_inner->setType() == Data::StickersType::Stickers) {
 | |
| 			auto &stickers = _session->data().stickers();
 | |
| 			stickers.notifyStickerSetInstalled(setId);
 | |
| 		}
 | |
| 		closeBox();
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	_inner->errors(
 | |
| 	) | rpl::start_with_next([=](Error error) {
 | |
| 		handleError(error);
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	_inner->setArchived(
 | |
| 	) | rpl::start_with_next([=](uint64 setId) {
 | |
| 		const auto type = _inner->setType();
 | |
| 		if (type == Data::StickersType::Emoji) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		showToast((type == Data::StickersType::Masks)
 | |
| 				? tr::lng_masks_has_been_archived(tr::now)
 | |
| 				: tr::lng_stickers_has_been_archived(tr::now));
 | |
| 
 | |
| 		auto &order = (type == Data::StickersType::Masks)
 | |
| 			? _session->data().stickers().maskSetsOrderRef()
 | |
| 			: _session->data().stickers().setsOrderRef();
 | |
| 		const auto index = order.indexOf(setId);
 | |
| 		if (index != -1) {
 | |
| 			order.removeAt(index);
 | |
| 
 | |
| 			auto &local = _session->local();
 | |
| 			if (type == Data::StickersType::Masks) {
 | |
| 				local.writeInstalledMasks();
 | |
| 				local.writeArchivedMasks();
 | |
| 			} else {
 | |
| 				local.writeInstalledStickers();
 | |
| 				local.writeArchivedStickers();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		_session->data().stickers().notifyUpdated(type);
 | |
| 
 | |
| 		closeBox();
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void StickerSetBox::addStickers() {
 | |
| 	_inner->install();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::copyStickersLink() {
 | |
| 	const auto part = _inner->isEmojiSet() ? u"addemoji"_q : "addstickers";
 | |
| 	const auto url = _session->createInternalLinkFull(
 | |
| 		part + '/' + _inner->shortName());
 | |
| 	QGuiApplication::clipboard()->setText(url);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::copyTitle() {
 | |
| 	_inner->title(
 | |
| 	) | rpl::start_with_next([this](const TextWithEntities &value) {
 | |
| 		QGuiApplication::clipboard()->setText(value.text);
 | |
| 		_show->showBox(
 | |
| 			Ui::MakeInformBox(ktr("ktg_stickers_title_copied")));
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void StickerSetBox::handleError(Error error) {
 | |
| 	const auto guard = gsl::finally(crl::guard(this, [=] {
 | |
| 		closeBox();
 | |
| 	}));
 | |
| 
 | |
| 	switch (error) {
 | |
| 	case Error::NotFound:
 | |
| 		_show->showBox(
 | |
| 			Ui::MakeInformBox(tr::lng_stickers_not_found(tr::now)));
 | |
| 		break;
 | |
| 	default: Unexpected("Error in StickerSetBox::handleError.");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerSetBox::updateTitleAndButtons() {
 | |
| 	setTitle(_inner->title());
 | |
| 	updateButtons();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::updateButtons() {
 | |
| 	clearButtons();
 | |
| 	if (_inner->loaded()) {
 | |
| 		const auto type = _inner->setType();
 | |
| 		const auto share = [=] {
 | |
| 			copyStickersLink();
 | |
| 			showToast(type == Data::StickersType::Emoji
 | |
| 					? tr::lng_stickers_copied_emoji(tr::now)
 | |
| 					: tr::lng_stickers_copied(tr::now));
 | |
| 		};
 | |
| 		const auto moreButton = addTopButton(st::infoTopBarMenu);
 | |
| 		moreButton->setClickedCallback([=] { showMenu(moreButton.data()); });
 | |
| 
 | |
| 		if (_inner->notInstalled()) {
 | |
| 			if (!_session->premium()
 | |
| 				&& _session->premiumPossible()
 | |
| 				&& _inner->premiumEmojiSet()) {
 | |
| 				const auto &st = st::premiumPreviewDoubledLimitsBox;
 | |
| 				setStyle(st);
 | |
| 				auto button = CreateUnlockButton(
 | |
| 					this,
 | |
| 					tr::lng_premium_unlock_emoji());
 | |
| 				button->resizeToWidth(st::boxWideWidth
 | |
| 					- st.buttonPadding.left()
 | |
| 					- st.buttonPadding.left());
 | |
| 				button->setClickedCallback([=] {
 | |
| 					using namespace ChatHelpers;
 | |
| 					const auto usage = WindowUsage::PremiumPromo;
 | |
| 					if (const auto window = _show->resolveWindow(usage)) {
 | |
| 						Settings::ShowPremium(window, u"animated_emoji"_q);
 | |
| 					}
 | |
| 				});
 | |
| 				addButton(std::move(button));
 | |
| 			} else {
 | |
| 				auto addText = (type == Data::StickersType::Emoji)
 | |
| 					? tr::lng_stickers_add_emoji()
 | |
| 					: (type == Data::StickersType::Masks)
 | |
| 					? tr::lng_stickers_add_masks()
 | |
| 					: tr::lng_stickers_add_pack();
 | |
| 				addButton(std::move(addText), [=] { addStickers(); });
 | |
| 				addButton(tr::lng_cancel(), [=] { closeBox(); });
 | |
| 			}
 | |
| 
 | |
| 			/*
 | |
| 			if (!_inner->shortName().isEmpty()) {
 | |
| 				const auto top = addTopButton(st::infoTopBarMenu);
 | |
| 				const auto menu
 | |
| 					= std::make_shared<base::unique_qptr<Ui::PopupMenu>>();
 | |
| 				top->setClickedCallback([=] {
 | |
| 					*menu = base::make_unique_q<Ui::PopupMenu>(
 | |
| 						top,
 | |
| 						st::popupMenuWithIcons);
 | |
| 					(*menu)->addAction(
 | |
| 						((type == Data::StickersType::Emoji)
 | |
| 							? tr::lng_stickers_share_emoji
 | |
| 							: (type == Data::StickersType::Masks)
 | |
| 							? tr::lng_stickers_share_masks
 | |
| 							: tr::lng_stickers_share_pack)(tr::now),
 | |
| 						[=] { share(); closeBox(); },
 | |
| 						&st::menuIconShare);
 | |
| 					(*menu)->popup(QCursor::pos());
 | |
| 					return true;
 | |
| 				});
 | |
| 			}
 | |
| 			*/
 | |
| 		} else if (_inner->official()) {
 | |
| 			addButton(tr::lng_about_done(), [=] { closeBox(); });
 | |
| 		} else {
 | |
| 			auto shareText = (type == Data::StickersType::Emoji)
 | |
| 				? tr::lng_stickers_share_emoji()
 | |
| 				: (type == Data::StickersType::Masks)
 | |
| 				? tr::lng_stickers_share_masks()
 | |
| 				: tr::lng_stickers_share_pack();
 | |
| 			addButton(std::move(shareText), std::move(share));
 | |
| 			addButton(tr::lng_cancel(), [=] { closeBox(); });
 | |
| 
 | |
| 			/*
 | |
| 			if (!_inner->shortName().isEmpty()) {
 | |
| 				const auto top = addTopButton(st::infoTopBarMenu);
 | |
| 				const auto archive = [=] {
 | |
| 					_inner->archiveStickers();
 | |
| 				};
 | |
| 				const auto remove = [=] {
 | |
| 					const auto session = &_show->session();
 | |
| 					auto box = ChatHelpers::MakeConfirmRemoveSetBox(
 | |
| 						session,
 | |
| 						st::boxLabel,
 | |
| 						_inner->setId());
 | |
| 					if (box) {
 | |
| 						_show->showBox(std::move(box));
 | |
| 					}
 | |
| 				};
 | |
| 				const auto menu
 | |
| 					= std::make_shared<base::unique_qptr<Ui::PopupMenu>>();
 | |
| 				top->setClickedCallback([=] {
 | |
| 					*menu = base::make_unique_q<Ui::PopupMenu>(
 | |
| 						top,
 | |
| 						st::popupMenuWithIcons);
 | |
| 					if (type == Data::StickersType::Emoji) {
 | |
| 						(*menu)->addAction(
 | |
| 							tr::lng_custom_emoji_remove_pack_button(tr::now),
 | |
| 							remove,
 | |
| 							&st::menuIconRemove);
 | |
| 					} else {
 | |
| 						(*menu)->addAction(
 | |
| 							(type == Data::StickersType::Masks
 | |
| 								? tr::lng_masks_archive_pack(tr::now)
 | |
| 								: tr::lng_stickers_archive_pack(tr::now)),
 | |
| 							archive,
 | |
| 							&st::menuIconArchive);
 | |
| 					}
 | |
| 					(*menu)->popup(QCursor::pos());
 | |
| 					return true;
 | |
| 				});
 | |
| 			}
 | |
| 			*/
 | |
| 		}
 | |
| 	} else {
 | |
| 		addButton(tr::lng_cancel(), [=] { closeBox(); });
 | |
| 	}
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| bool StickerSetBox::showMenu(not_null<Ui::IconButton*> button) {
 | |
| 	if (_menu) {
 | |
| 		_menu->hideAnimated(Ui::InnerDropdown::HideOption::IgnoreShow);
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	_menu = base::make_unique_q<Ui::DropdownMenu>(
 | |
| 		window(),
 | |
| 		st::dropdownMenuWithIcons);
 | |
| 	const auto weak = _menu.get();
 | |
| 	_menu->setHiddenCallback([=] {
 | |
| 		weak->deleteLater();
 | |
| 		if (_menu == weak) {
 | |
| 			button->setForceRippled(false);
 | |
| 		}
 | |
| 	});
 | |
| 	_menu->setShowStartCallback([=] {
 | |
| 		if (_menu == weak) {
 | |
| 			button->setForceRippled(true);
 | |
| 		}
 | |
| 	});
 | |
| 	_menu->setHideStartCallback([=] {
 | |
| 		if (_menu == weak) {
 | |
| 			button->setForceRippled(false);
 | |
| 		}
 | |
| 	});
 | |
| 	button->installEventFilter(_menu);
 | |
| 
 | |
| 	_menu->addAction(
 | |
| 		ktr("ktg_stickers_copy_title"),
 | |
| 		[=] { copyTitle(); },
 | |
| 		&st::menuIconCopy);
 | |
| 
 | |
| 	if (!_inner->shortName().isEmpty()) {
 | |
| 		_menu->addAction(
 | |
| 			tr::lng_stickers_share_pack(tr::now),
 | |
| 			[=] { copyStickersLink(); },
 | |
| 			&st::menuIconShare);
 | |
| 	}
 | |
| 
 | |
| 	if (!_inner->notInstalled()) {
 | |
| 		const auto archive = [=] {
 | |
| 			_inner->archiveStickers();
 | |
| 			closeBox();
 | |
| 		};
 | |
| 		_menu->addAction(
 | |
| 			tr::lng_stickers_archive_pack(tr::now),
 | |
| 			archive,
 | |
| 			&st::menuIconArchive);
 | |
| 	}
 | |
| 
 | |
| 	const auto parentTopLeft = window()->mapToGlobal(QPoint());
 | |
| 	const auto buttonTopLeft = button->mapToGlobal(QPoint());
 | |
| 	const auto parentRect = QRect(parentTopLeft, window()->size());
 | |
| 	const auto buttonRect = QRect(buttonTopLeft, button->size());
 | |
| 	_menu->move(
 | |
| 		buttonRect.x() + buttonRect.width() - _menu->width() - parentRect.x(),
 | |
| 		buttonRect.y() + buttonRect.height() - parentRect.y() - style::ConvertScale(18));
 | |
| 	_menu->showAnimated(Ui::PanelAnimation::Origin::TopRight);
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void StickerSetBox::resizeEvent(QResizeEvent *e) {
 | |
| 	BoxContent::resizeEvent(e);
 | |
| 	_inner->resize(width(), _inner->height());
 | |
| }
 | |
| 
 | |
| StickerSetBox::Inner::Inner(
 | |
| 	QWidget *parent,
 | |
| 	std::shared_ptr<ChatHelpers::Show> show,
 | |
| 	const StickerSetIdentifier &set,
 | |
| 	Data::StickersType type)
 | |
| : RpWidget(parent)
 | |
| , _show(std::move(show))
 | |
| , _session(&_show->session())
 | |
| , _api(&_session->mtp())
 | |
| , _setId(set.id)
 | |
| , _setAccessHash(set.accessHash)
 | |
| , _setShortName(set.shortName)
 | |
| , _pathGradient(std::make_unique<Ui::PathShiftGradient>(
 | |
| 	st::windowBgRipple,
 | |
| 	st::windowBgOver,
 | |
| 	[=] { repaintItems(); }))
 | |
| , _premiumMark(_session)
 | |
| , _updateItemsTimer([=] { updateItems(); })
 | |
| , _input(set)
 | |
| , _padding((type == Data::StickersType::Emoji)
 | |
| 	? st::emojiSetPadding
 | |
| 	: st::stickersPadding)
 | |
| , _previewTimer([=] { showPreview(); }) {
 | |
| 	setAttribute(Qt::WA_OpaquePaintEvent);
 | |
| 
 | |
| 	_api.request(MTPmessages_GetStickerSet(
 | |
| 		Data::InputStickerSet(_input),
 | |
| 		MTP_int(0) // hash
 | |
| 	)).done([=](const MTPmessages_StickerSet &result) {
 | |
| 		gotSet(result);
 | |
| 	}).fail([=] {
 | |
| 		_loaded = true;
 | |
| 		_errors.fire(Error::NotFound);
 | |
| 	}).send();
 | |
| 
 | |
| 	_session->api().updateStickers();
 | |
| 
 | |
| 	_session->downloaderTaskFinished(
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		updateItems();
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	setMouseTracking(true);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
 | |
| 	_pack.clear();
 | |
| 	_emoji.clear();
 | |
| 	_elements.clear();
 | |
| 	_selected = -1;
 | |
| 	setCursor(style::cur_default);
 | |
| 	const auto owner = &_session->data();
 | |
| 	const auto premiumPossible = _session->premiumPossible();
 | |
| 	set.match([&](const MTPDmessages_stickerSet &data) {
 | |
| 		const auto &v = data.vdocuments().v;
 | |
| 		_pack.reserve(v.size());
 | |
| 		_elements.reserve(v.size());
 | |
| 		for (const auto &item : v) {
 | |
| 			const auto document = owner->processDocument(item);
 | |
| 			const auto sticker = document->sticker();
 | |
| 			if (!sticker) {
 | |
| 				continue;
 | |
| 			}
 | |
| 			_pack.push_back(document);
 | |
| 			if (!document->isPremiumSticker() || premiumPossible) {
 | |
| 				_elements.push_back({
 | |
| 					document,
 | |
| 					document->createMediaView(),
 | |
| 				});
 | |
| 			}
 | |
| 		}
 | |
| 		for (const auto &pack : data.vpacks().v) {
 | |
| 			pack.match([&](const MTPDstickerPack &pack) {
 | |
| 				if (const auto emoji = Ui::Emoji::Find(qs(pack.vemoticon()))) {
 | |
| 					const auto original = emoji->original();
 | |
| 					auto &stickers = pack.vdocuments().v;
 | |
| 
 | |
| 					auto p = StickersPack();
 | |
| 					p.reserve(stickers.size());
 | |
| 					for (auto j = 0, c = int(stickers.size()); j != c; ++j) {
 | |
| 						auto doc = _session->data().document(stickers[j].v);
 | |
| 						if (!doc || !doc->sticker()) continue;
 | |
| 
 | |
| 						p.push_back(doc);
 | |
| 					}
 | |
| 					_emoji[original] = std::move(p);
 | |
| 				}
 | |
| 			});
 | |
| 		}
 | |
| 		data.vset().match([&](const MTPDstickerSet &set) {
 | |
| 			_setTitle = _session->data().stickers().getSetTitle(
 | |
| 				set);
 | |
| 			_setShortName = qs(set.vshort_name());
 | |
| 			_setId = set.vid().v;
 | |
| 			_setAccessHash = set.vaccess_hash().v;
 | |
| 			_setHash = set.vhash().v;
 | |
| 			_setCount = set.vcount().v;
 | |
| 			_setFlags = Data::ParseStickersSetFlags(set);
 | |
| 			_setInstallDate = set.vinstalled_date().value_or(0);
 | |
| 			_setThumbnailDocumentId = set.vthumb_document_id().value_or_empty();
 | |
| 			_setThumbnail = [&] {
 | |
| 				if (const auto thumbs = set.vthumbs()) {
 | |
| 					for (const auto &thumb : thumbs->v) {
 | |
| 						const auto result = Images::FromPhotoSize(
 | |
| 							_session,
 | |
| 							set,
 | |
| 							thumb);
 | |
| 						if (result.location.valid()) {
 | |
| 							_setThumbnailType
 | |
| 								= Data::ThumbnailTypeFromPhotoSize(thumb);
 | |
| 							return result;
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				return ImageWithLocation();
 | |
| 			}();
 | |
| 			const auto &sets = _session->data().stickers().sets();
 | |
| 			const auto it = sets.find(_setId);
 | |
| 			if (it != sets.cend()) {
 | |
| 				const auto set = it->second.get();
 | |
| 				const auto clientFlags = set->flags
 | |
| 					& (SetFlag::Featured
 | |
| 						| SetFlag::NotLoaded
 | |
| 						| SetFlag::Unread
 | |
| 						| SetFlag::Special);
 | |
| 				_setFlags |= clientFlags;
 | |
| 				set->flags = _setFlags;
 | |
| 				set->installDate = _setInstallDate;
 | |
| 				set->stickers = _pack;
 | |
| 				set->emoji = _emoji;
 | |
| 				set->setThumbnail(_setThumbnail, _setThumbnailType);
 | |
| 			}
 | |
| 		});
 | |
| 	}, [&](const MTPDmessages_stickerSetNotModified &data) {
 | |
| 		LOG(("API Error: Unexpected messages.stickerSetNotModified."));
 | |
| 	});
 | |
| 
 | |
| 	if (_pack.isEmpty()) {
 | |
| 		_errors.fire(Error::NotFound);
 | |
| 		return;
 | |
| 	}
 | |
| 	_perRow = isEmojiSet() ? kEmojiPerRow : kStickersPerRow;
 | |
| 	_rowsCount = (_pack.size() + _perRow - 1) / _perRow;
 | |
| 	_singleSize = isEmojiSet() ? st::emojiSetSize : st::stickersSize;
 | |
| 
 | |
| 	resize(
 | |
| 		_padding.left() + _perRow * _singleSize.width(),
 | |
| 		_padding.top() + _rowsCount * _singleSize.height() + _padding.bottom());
 | |
| 
 | |
| 	_loaded = true;
 | |
| 	updateSelected();
 | |
| 	_updateControls.fire({});
 | |
| }
 | |
| 
 | |
| rpl::producer<uint64> StickerSetBox::Inner::setInstalled() const {
 | |
| 	return _setInstalled.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<uint64> StickerSetBox::Inner::setArchived() const {
 | |
| 	return _setArchived.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<> StickerSetBox::Inner::updateControls() const {
 | |
| 	return _updateControls.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<StickerSetBox::Error> StickerSetBox::Inner::errors() const {
 | |
| 	return _errors.events();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::installDone(
 | |
| 		const MTPmessages_StickerSetInstallResult &result) {
 | |
| 	auto &stickers = _session->data().stickers();
 | |
| 	auto &sets = stickers.setsRef();
 | |
| 	const auto type = setType();
 | |
| 
 | |
| 	const bool wasArchived = (_setFlags & SetFlag::Archived);
 | |
| 	if (wasArchived && type != Data::StickersType::Emoji) {
 | |
| 		const auto index = ((type == Data::StickersType::Masks)
 | |
| 			? stickers.archivedMaskSetsOrderRef()
 | |
| 			: stickers.archivedSetsOrderRef()).indexOf(_setId);
 | |
| 		if (index >= 0) {
 | |
| 			((type == Data::StickersType::Masks)
 | |
| 				? stickers.archivedMaskSetsOrderRef()
 | |
| 				: stickers.archivedSetsOrderRef()).removeAt(index);
 | |
| 		}
 | |
| 	}
 | |
| 	_setInstallDate = base::unixtime::now();
 | |
| 	_setFlags &= ~SetFlag::Archived;
 | |
| 	_setFlags |= SetFlag::Installed;
 | |
| 	auto it = sets.find(_setId);
 | |
| 	if (it == sets.cend()) {
 | |
| 		it = sets.emplace(
 | |
| 			_setId,
 | |
| 			std::make_unique<StickersSet>(
 | |
| 				&_session->data(),
 | |
| 				_setId,
 | |
| 				_setAccessHash,
 | |
| 				_setHash,
 | |
| 				_setTitle,
 | |
| 				_setShortName,
 | |
| 				_setCount,
 | |
| 				_setFlags,
 | |
| 				_setInstallDate)).first;
 | |
| 	} else {
 | |
| 		it->second->flags = _setFlags;
 | |
| 		it->second->installDate = _setInstallDate;
 | |
| 	}
 | |
| 	const auto set = it->second.get();
 | |
| 	set->thumbnailDocumentId = _setThumbnailDocumentId;
 | |
| 	set->setThumbnail(_setThumbnail, _setThumbnailType);
 | |
| 	set->stickers = _pack;
 | |
| 	set->emoji = _emoji;
 | |
| 
 | |
| 	auto &order = (type == Data::StickersType::Emoji)
 | |
| 		? stickers.emojiSetsOrderRef()
 | |
| 		: (type == Data::StickersType::Masks)
 | |
| 		? stickers.maskSetsOrderRef()
 | |
| 		: stickers.setsOrderRef();
 | |
| 	const auto insertAtIndex = 0, currentIndex = int(order.indexOf(_setId));
 | |
| 	if (currentIndex != insertAtIndex) {
 | |
| 		if (currentIndex > 0) {
 | |
| 			order.removeAt(currentIndex);
 | |
| 		}
 | |
| 		order.insert(insertAtIndex, _setId);
 | |
| 	}
 | |
| 
 | |
| 	const auto customIt = sets.find(Data::Stickers::CustomSetId);
 | |
| 	if (customIt != sets.cend()) {
 | |
| 		const auto custom = customIt->second.get();
 | |
| 		for (const auto sticker : std::as_const(_pack)) {
 | |
| 			const int removeIndex = custom->stickers.indexOf(sticker);
 | |
| 			if (removeIndex >= 0) {
 | |
| 				custom->stickers.removeAt(removeIndex);
 | |
| 			}
 | |
| 		}
 | |
| 		if (custom->stickers.isEmpty()) {
 | |
| 			sets.erase(customIt);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (result.type() == mtpc_messages_stickerSetInstallResultArchive) {
 | |
| 		stickers.applyArchivedResult(
 | |
| 			result.c_messages_stickerSetInstallResultArchive());
 | |
| 	} else {
 | |
| 		auto &storage = _session->local();
 | |
| 		if (wasArchived && type != Data::StickersType::Emoji) {
 | |
| 			if (type == Data::StickersType::Masks) {
 | |
| 				storage.writeArchivedMasks();
 | |
| 			} else {
 | |
| 				storage.writeArchivedStickers();
 | |
| 			}
 | |
| 		}
 | |
| 		if (type == Data::StickersType::Emoji) {
 | |
| 			storage.writeInstalledCustomEmoji();
 | |
| 		} else if (type == Data::StickersType::Masks) {
 | |
| 			storage.writeInstalledMasks();
 | |
| 		} else {
 | |
| 			storage.writeInstalledStickers();
 | |
| 		}
 | |
| 		stickers.notifyUpdated(type);
 | |
| 	}
 | |
| 	_setInstalled.fire_copy(_setId);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::mousePressEvent(QMouseEvent *e) {
 | |
| 	if (e->button() != Qt::LeftButton) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto index = stickerFromGlobalPos(e->globalPos());
 | |
| 	if (index < 0 || index >= _pack.size()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_previewTimer.callOnce(QApplication::startDragTime());
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::mouseMoveEvent(QMouseEvent *e) {
 | |
| 	updateSelected();
 | |
| 	if (_previewShown >= 0) {
 | |
| 		showPreviewAt(e->globalPos());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::showPreviewAt(QPoint globalPos) {
 | |
| 	const auto index = stickerFromGlobalPos(globalPos);
 | |
| 	if (index >= 0
 | |
| 		&& index < _pack.size()
 | |
| 		&& index != _previewShown) {
 | |
| 		_previewShown = index;
 | |
| 		_show->showMediaPreview(
 | |
| 			Data::FileOriginStickerSet(_setId, _setAccessHash),
 | |
| 			_pack[_previewShown]);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::leaveEventHook(QEvent *e) {
 | |
| 	setSelected(-1);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::mouseReleaseEvent(QMouseEvent *e) {
 | |
| 	if (_previewShown >= 0) {
 | |
| 		_previewShown = -1;
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!_previewTimer.isActive()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_previewTimer.cancel();
 | |
| 	const auto index = stickerFromGlobalPos(e->globalPos());
 | |
| 	if (index < 0 || index >= _pack.size()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	chosen(index, _pack[index], {});
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::chosen(
 | |
| 		int index,
 | |
| 		not_null<DocumentData*> sticker,
 | |
| 		Api::SendOptions options) {
 | |
| 	const auto animation = options.scheduled
 | |
| 		? Ui::MessageSendingAnimationFrom()
 | |
| 		: messageSentAnimationInfo(index, sticker);
 | |
| 	_show->processChosenSticker({
 | |
| 		.document = sticker,
 | |
| 		.options = options,
 | |
| 		.messageSendingFrom = animation,
 | |
| 	});
 | |
| }
 | |
| 
 | |
| auto StickerSetBox::Inner::messageSentAnimationInfo(
 | |
| 	int index,
 | |
| 	not_null<DocumentData*> document) const
 | |
| -> Ui::MessageSendingAnimationFrom {
 | |
| 	if (index < 0 || index >= _pack.size() || _pack[index] != document) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	const auto row = index / _perRow;
 | |
| 	const auto column = index % _perRow;
 | |
| 	const auto left = _padding.left() + column * _singleSize.width();
 | |
| 	const auto top = _padding.top() + row * _singleSize.height();
 | |
| 	const auto rect = QRect(QPoint(left, top), _singleSize);
 | |
| 	const auto size = ChatHelpers::ComputeStickerSize(
 | |
| 		document,
 | |
| 		boundingBoxSize());
 | |
| 	const auto innerPos = QPoint(
 | |
| 		(rect.width() - size.width()) / 2,
 | |
| 		(rect.height() - size.height()) / 2);
 | |
| 	return {
 | |
| 		.type = Ui::MessageSendingAnimationFrom::Type::Sticker,
 | |
| 		.localId = _session->data().nextLocalMessageId(),
 | |
| 		.globalStartGeometry = mapToGlobal(
 | |
| 			QRect(rect.topLeft() + innerPos, size)),
 | |
| 	};
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::contextMenuEvent(QContextMenuEvent *e) {
 | |
| 	const auto index = stickerFromGlobalPos(e->globalPos());
 | |
| 	if (index < 0
 | |
| 		|| index >= _pack.size()
 | |
| 		|| setType() == Data::StickersType::Masks) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_previewTimer.cancel();
 | |
| 	_menu = base::make_unique_q<Ui::PopupMenu>(
 | |
| 		this,
 | |
| 		st::popupMenuWithIcons);
 | |
| 	const auto type = _show->sendMenuType();
 | |
| 	if (setType() == Data::StickersType::Emoji) {
 | |
| 		if (const auto t = PrepareTextFromEmoji(_pack[index]); !t.empty()) {
 | |
| 			_menu->addAction(tr::lng_mediaview_copy(tr::now), [=] {
 | |
| 				if (auto data = TextUtilities::MimeDataFromText(t)) {
 | |
| 					QGuiApplication::clipboard()->setMimeData(data.release());
 | |
| 				}
 | |
| 			}, &st::menuIconCopy);
 | |
| 		}
 | |
| 	} else if (type != SendMenu::Type::Disabled) {
 | |
| 		const auto document = _pack[index];
 | |
| 		const auto sendSelected = [=](Api::SendOptions options) {
 | |
| 			chosen(index, document, options);
 | |
| 		};
 | |
| 		SendMenu::FillSendMenu(
 | |
| 			_menu.get(),
 | |
| 			type,
 | |
| 			SendMenu::DefaultSilentCallback(sendSelected),
 | |
| 			SendMenu::DefaultScheduleCallback(_show, type, sendSelected),
 | |
| 			SendMenu::DefaultWhenOnlineCallback(sendSelected));
 | |
| 
 | |
| 		const auto show = _show;
 | |
| 		const auto toggleFavedSticker = [=] {
 | |
| 			Api::ToggleFavedSticker(
 | |
| 				show,
 | |
| 				document,
 | |
| 				Data::FileOriginStickerSet(Data::Stickers::FavedSetId, 0));
 | |
| 		};
 | |
| 		const auto isFaved = document->owner().stickers().isFaved(document);
 | |
| 		_menu->addAction(
 | |
| 			(isFaved
 | |
| 				? tr::lng_faved_stickers_remove
 | |
| 				: tr::lng_faved_stickers_add)(tr::now),
 | |
| 			toggleFavedSticker,
 | |
| 			(isFaved
 | |
| 				? &st::menuIconUnfave
 | |
| 				: &st::menuIconFave));
 | |
| 	}
 | |
| 	if (_menu->empty()) {
 | |
| 		_menu = nullptr;
 | |
| 	} else {
 | |
| 		_menu->popup(QCursor::pos());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::updateSelected() {
 | |
| 	auto selected = stickerFromGlobalPos(QCursor::pos());
 | |
| 	setSelected(setType() == Data::StickersType::Masks ? -1 : selected);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::setSelected(int selected) {
 | |
| 	if (_selected != selected) {
 | |
| 		startOverAnimation(_selected, 1., 0.);
 | |
| 		_selected = selected;
 | |
| 		startOverAnimation(_selected, 0., 1.);
 | |
| 		setCursor(_selected >= 0 ? style::cur_pointer : style::cur_default);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::startOverAnimation(int index, float64 from, float64 to) {
 | |
| 	if (index < 0 || index >= _elements.size()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_elements[index].overAnimation.start([=] {
 | |
| 		const auto row = index / _perRow;
 | |
| 		const auto column = index % _perRow;
 | |
| 		const auto left = _padding.left() + column * _singleSize.width();
 | |
| 		const auto top = _padding.top() + row * _singleSize.height();
 | |
| 		rtlupdate(left, top, _singleSize.width(), _singleSize.height());
 | |
| 	}, from, to, st::emojiPanDuration);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::showPreview() {
 | |
| 	_previewShown = -1;
 | |
| 	showPreviewAt(QCursor::pos());
 | |
| }
 | |
| 
 | |
| not_null<Lottie::MultiPlayer*> StickerSetBox::Inner::getLottiePlayer() {
 | |
| 	if (!_lottiePlayer) {
 | |
| 		_lottiePlayer = std::make_unique<Lottie::MultiPlayer>(
 | |
| 			Lottie::Quality::Default,
 | |
| 			Lottie::MakeFrameRenderer());
 | |
| 		_lottiePlayer->updates(
 | |
| 		) | rpl::start_with_next([=] {
 | |
| 			updateItems();
 | |
| 		}, lifetime());
 | |
| 	}
 | |
| 	return _lottiePlayer.get();
 | |
| }
 | |
| 
 | |
| int32 StickerSetBox::Inner::stickerFromGlobalPos(const QPoint &p) const {
 | |
| 	QPoint l(mapFromGlobal(p));
 | |
| 	if (rtl()) l.setX(width() - l.x());
 | |
| 	int32 row = (l.y() >= _padding.top()) ? qFloor((l.y() - _padding.top()) / _singleSize.height()) : -1;
 | |
| 	int32 col = (l.x() >= _padding.left()) ? qFloor((l.x() - _padding.left()) / _singleSize.width()) : -1;
 | |
| 	if (row >= 0 && col >= 0 && col < _perRow) {
 | |
| 		int32 result = row * _perRow + col;
 | |
| 		return (result < _pack.size()) ? result : -1;
 | |
| 	}
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::paintEvent(QPaintEvent *e) {
 | |
| 	Painter p(this);
 | |
| 
 | |
| 	_repaintScheduled = false;
 | |
| 
 | |
| 	p.fillRect(e->rect(), st::boxBg);
 | |
| 	if (_elements.empty()) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	int32 from = qFloor(e->rect().top() / _singleSize.height()), to = qFloor(e->rect().bottom() / _singleSize.height()) + 1;
 | |
| 
 | |
| 	_pathGradient->startFrame(0, width(), width() / 2);
 | |
| 
 | |
| 	const auto now = crl::now();
 | |
| 	const auto paused = On(PowerSaving::kStickersPanel)
 | |
| 		|| _show->paused(ChatHelpers::PauseReason::Layer);
 | |
| 	for (int32 i = from; i < to; ++i) {
 | |
| 		for (int32 j = 0; j < _perRow; ++j) {
 | |
| 			int32 index = i * _perRow + j;
 | |
| 			if (index >= _elements.size()) {
 | |
| 				break;
 | |
| 			}
 | |
| 			const auto pos = QPoint(
 | |
| 				_padding.left() + j * _singleSize.width(),
 | |
| 				_padding.top() + i * _singleSize.height());
 | |
| 			paintSticker(p, index, pos, paused, now);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (_lottiePlayer && !paused) {
 | |
| 		_lottiePlayer->markFrameShown();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool StickerSetBox::Inner::isEmojiSet() const {
 | |
| 	return (_setFlags & Data::StickersSetFlag::Emoji);
 | |
| }
 | |
| 
 | |
| uint64 StickerSetBox::Inner::setId() const {
 | |
| 	return _setId;
 | |
| }
 | |
| 
 | |
| QSize StickerSetBox::Inner::boundingBoxSize() const {
 | |
| 	if (isEmojiSet()) {
 | |
| 		using namespace Data;
 | |
| 		const auto size = FrameSizeFromTag(CustomEmojiSizeTag::Large)
 | |
| 			/ style::DevicePixelRatio();
 | |
| 		return { size, size };
 | |
| 	}
 | |
| 	return QSize(
 | |
| 		_singleSize.width() - st::roundRadiusSmall * 2,
 | |
| 		_singleSize.height() - st::roundRadiusSmall * 2);
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::visibleTopBottomUpdated(
 | |
| 		int visibleTop,
 | |
| 		int visibleBottom) {
 | |
| 	if (_visibleTop != visibleTop || _visibleBottom != visibleBottom) {
 | |
| 		_visibleTop = visibleTop;
 | |
| 		_visibleBottom = visibleBottom;
 | |
| 		_lastScrolledAt = crl::now();
 | |
| 		update();
 | |
| 	}
 | |
| 	const auto pauseInRows = [&](int fromRow, int tillRow) {
 | |
| 		Expects(fromRow <= tillRow);
 | |
| 
 | |
| 		for (auto i = fromRow; i != tillRow; ++i) {
 | |
| 			for (auto j = 0; j != _perRow; ++j) {
 | |
| 				const auto index = i * _perRow + j;
 | |
| 				if (index >= _elements.size()) {
 | |
| 					break;
 | |
| 				}
 | |
| 				if (const auto lottie = _elements[index].lottie) {
 | |
| 					_lottiePlayer->pause(lottie);
 | |
| 				} else if (auto &webm = _elements[index].webm) {
 | |
| 					webm = nullptr;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	};
 | |
| 	const auto rowsTop = _padding.top();
 | |
| 	const auto singleHeight = _singleSize.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];
 | |
| 
 | |
| 	element.lottie = ChatHelpers::LottieAnimationFromDocument(
 | |
| 		getLottiePlayer(),
 | |
| 		element.documentMedia.get(),
 | |
| 		ChatHelpers::StickerLottieSize::StickerSet,
 | |
| 		boundingBoxSize() * style::DevicePixelRatio());
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::setupWebm(int index) {
 | |
| 	auto &element = _elements[index];
 | |
| 
 | |
| 	const auto document = element.document;
 | |
| 	auto callback = [=](Media::Clip::Notification notification) {
 | |
| 		clipCallback(notification, document, index);
 | |
| 	};
 | |
| 	element.webm = Media::Clip::MakeReader(
 | |
| 		element.documentMedia->owner()->location(),
 | |
| 		element.documentMedia->bytes(),
 | |
| 		std::move(callback));
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::clipCallback(
 | |
| 		Media::Clip::Notification notification,
 | |
| 		not_null<DocumentData*> document,
 | |
| 		int index) {
 | |
| 	const auto i = (index < _elements.size()
 | |
| 		&& _elements[index].document == document)
 | |
| 		? (_elements.begin() + index)
 | |
| 		: ranges::find(_elements, document, &Element::document);
 | |
| 	if (i == end(_elements)) {
 | |
| 		return;
 | |
| 	}
 | |
| 	using namespace Media::Clip;
 | |
| 	switch (notification) {
 | |
| 	case Notification::Reinit: {
 | |
| 		auto &webm = i->webm;
 | |
| 		if (webm->state() == State::Error) {
 | |
| 			webm.setBad();
 | |
| 		} else if (webm->ready() && !webm->started()) {
 | |
| 			const auto size = ChatHelpers::ComputeStickerSize(
 | |
| 				i->document,
 | |
| 				boundingBoxSize());
 | |
| 			webm->start({ .frame = size, .keepAlpha = true });
 | |
| 		}
 | |
| 	} break;
 | |
| 
 | |
| 	case Notification::Repaint: break;
 | |
| 	}
 | |
| 
 | |
| 	updateItems();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::setupEmoji(int index) {
 | |
| 	auto &element = _elements[index];
 | |
| 	element.emoji = resolveCustomEmoji(element.document);
 | |
| }
 | |
| 
 | |
| not_null<Ui::Text::CustomEmoji*> StickerSetBox::Inner::resolveCustomEmoji(
 | |
| 		not_null<DocumentData*> document) {
 | |
| 	const auto i = _customEmoji.find(document);
 | |
| 	if (i != end(_customEmoji)) {
 | |
| 		return i->second.get();
 | |
| 	}
 | |
| 	auto emoji = document->session().data().customEmojiManager().create(
 | |
| 		document,
 | |
| 		[=] { customEmojiRepaint(); },
 | |
| 		Data::CustomEmojiManager::SizeTag::Large);
 | |
| 	return _customEmoji.emplace(
 | |
| 		document,
 | |
| 		std::move(emoji)
 | |
| 	).first->second.get();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::customEmojiRepaint() {
 | |
| 	if (_repaintScheduled) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_repaintScheduled = true;
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::paintSticker(
 | |
| 		Painter &p,
 | |
| 		int index,
 | |
| 		QPoint position,
 | |
| 		bool paused,
 | |
| 		crl::time now) const {
 | |
| 	if (const auto over = _elements[index].overAnimation.value((index == _selected) ? 1. : 0.)) {
 | |
| 		p.setOpacity(over);
 | |
| 		auto tl = position;
 | |
| 		if (rtl()) tl.setX(width() - tl.x() - _singleSize.width());
 | |
| 		Ui::FillRoundRect(p, QRect(tl, _singleSize), st::emojiPanHover, Ui::StickerHoverCorners);
 | |
| 		p.setOpacity(1);
 | |
| 	}
 | |
| 
 | |
| 	const auto &element = _elements[index];
 | |
| 	const auto document = element.document;
 | |
| 	const auto &media = element.documentMedia;
 | |
| 	const auto sticker = document->sticker();
 | |
| 	media->checkStickerSmall();
 | |
| 
 | |
| 	if (sticker->setType == Data::StickersType::Emoji) {
 | |
| 		const_cast<Inner*>(this)->setupEmoji(index);
 | |
| 	} else if (media->loaded()) {
 | |
| 		if (sticker->isLottie() && !element.lottie) {
 | |
| 			const_cast<Inner*>(this)->setupLottie(index);
 | |
| 		} else if (sticker->isWebm() && !element.webm) {
 | |
| 			const_cast<Inner*>(this)->setupWebm(index);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	const auto premium = document->isPremiumSticker();
 | |
| 	const auto size = ChatHelpers::ComputeStickerSize(
 | |
| 		document,
 | |
| 		boundingBoxSize());
 | |
| 	const auto ppos = position + QPoint(
 | |
| 		(_singleSize.width() - size.width()) / 2,
 | |
| 		(_singleSize.height() - size.height()) / 2);
 | |
| 	auto lottieFrame = QImage();
 | |
| 	if (element.emoji) {
 | |
| 		element.emoji->paint(p, {
 | |
| 			.textColor = st::windowFg->c,
 | |
| 			.now = now,
 | |
| 			.position = ppos,
 | |
| 			.paused = paused,
 | |
| 		});
 | |
| 	} else if (element.lottie && element.lottie->ready()) {
 | |
| 		lottieFrame = element.lottie->frame();
 | |
| 		p.drawImage(
 | |
| 			QRect(ppos, lottieFrame.size() / style::DevicePixelRatio()),
 | |
| 			lottieFrame);
 | |
| 
 | |
| 		_lottiePlayer->unpause(element.lottie);
 | |
| 	} else if (element.webm && element.webm->started()) {
 | |
| 		p.drawImage(ppos, element.webm->current({
 | |
| 			.frame = size,
 | |
| 			.keepAlpha = true,
 | |
| 		}, paused ? 0 : now));
 | |
| 	} else if (const auto image = media->getStickerSmall()) {
 | |
| 		const auto pixmap = image->pix(size);
 | |
| 		p.drawPixmapLeft(ppos, width(), pixmap);
 | |
| 		if (premium) {
 | |
| 			lottieFrame = pixmap.toImage().convertToFormat(
 | |
| 				QImage::Format_ARGB32_Premultiplied);
 | |
| 		}
 | |
| 	} else {
 | |
| 		ChatHelpers::PaintStickerThumbnailPath(
 | |
| 			p,
 | |
| 			media.get(),
 | |
| 			QRect(ppos, size),
 | |
| 			_pathGradient.get());
 | |
| 	}
 | |
| 	if (premium) {
 | |
| 		_premiumMark.paint(
 | |
| 			p,
 | |
| 			lottieFrame,
 | |
| 			element.premiumLock,
 | |
| 			position,
 | |
| 			_singleSize,
 | |
| 			width());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool StickerSetBox::Inner::loaded() const {
 | |
| 	return _loaded && !_pack.isEmpty();
 | |
| }
 | |
| 
 | |
| bool StickerSetBox::Inner::premiumEmojiSet() const {
 | |
| 	return (_setFlags & SetFlag::Emoji)
 | |
| 		&& !_pack.empty()
 | |
| 		&& _pack.front()->isPremiumEmoji();
 | |
| }
 | |
| 
 | |
| bool StickerSetBox::Inner::notInstalled() const {
 | |
| 	if (!_loaded) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	const auto &sets = _session->data().stickers().sets();
 | |
| 	const auto it = sets.find(_setId);
 | |
| 	if ((it == sets.cend())
 | |
| 		|| !(it->second->flags & SetFlag::Installed)
 | |
| 		|| (it->second->flags & SetFlag::Archived)) {
 | |
| 		return !_pack.empty();
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool StickerSetBox::Inner::official() const {
 | |
| 	return _loaded && _setShortName.isEmpty();
 | |
| }
 | |
| 
 | |
| rpl::producer<TextWithEntities> StickerSetBox::Inner::title() const {
 | |
| 	if (!_loaded) {
 | |
| 		return tr::lng_contacts_loading() | Ui::Text::ToWithEntities();
 | |
| 	} else if (_pack.isEmpty()) {
 | |
| 		return tr::lng_attach_failed() | Ui::Text::ToWithEntities();
 | |
| 	}
 | |
| 	auto text = TextWithEntities{ _setTitle };
 | |
| 	TextUtilities::ParseEntities(text, TextParseMentions);
 | |
| 	return rpl::single(text);
 | |
| }
 | |
| 
 | |
| QString StickerSetBox::Inner::shortName() const {
 | |
| 	return _setShortName;
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::install() {
 | |
| 	if (_installRequest) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_installRequest = _api.request(MTPmessages_InstallStickerSet(
 | |
| 		Data::InputStickerSet(_input),
 | |
| 		MTP_bool(false)
 | |
| 	)).done([=](const MTPmessages_StickerSetInstallResult &result) {
 | |
| 		installDone(result);
 | |
| 	}).fail([=] {
 | |
| 		_errors.fire(Error::NotFound);
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::archiveStickers() {
 | |
| 	_api.request(MTPmessages_InstallStickerSet(
 | |
| 		Data::InputStickerSet(_input),
 | |
| 		MTP_boolTrue()
 | |
| 	)).done([=](const MTPmessages_StickerSetInstallResult &result) {
 | |
| 		if (result.type() == mtpc_messages_stickerSetInstallResultSuccess) {
 | |
| 			_setArchived.fire_copy(_setId);
 | |
| 		}
 | |
| 	}).fail([=] {
 | |
| 		_show->showToast(Lang::Hard::ServerError());
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::updateItems() {
 | |
| 	const auto now = crl::now();
 | |
| 
 | |
| 	const auto delay = std::max(
 | |
| 		_lastScrolledAt + kMinAfterScrollDelay - now,
 | |
| 		_lastUpdatedAt + kMinRepaintDelay - now);
 | |
| 	if (delay <= 0) {
 | |
| 		repaintItems(now);
 | |
| 	} else if (!_updateItemsTimer.isActive()
 | |
| 		|| _updateItemsTimer.remainingTime() > kMinRepaintDelay) {
 | |
| 		_updateItemsTimer.callOnce(std::max(delay, kMinRepaintDelay));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void StickerSetBox::Inner::repaintItems(crl::time now) {
 | |
| 	_lastUpdatedAt = now ? now : crl::now();
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| StickerSetBox::Inner::~Inner() = default;
 |