tdesktop/Telegram/SourceFiles/data/stickers/data_custom_emoji.h

199 lines
5.2 KiB
C
Raw Normal View History

/*
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
*/
#pragma once
#include "data/stickers/data_stickers_set.h"
#include "ui/text/custom_emoji_instance.h"
2022-06-28 13:13:20 +00:00
#include "base/timer.h"
#include "base/weak_ptr.h"
namespace Main {
class Session;
} // namespace Main
namespace Data {
class Session;
class CustomEmojiLoader;
enum class CustomEmojiSizeTag : uchar {
Normal,
Large,
Isolated,
kCount,
};
2022-06-28 13:13:20 +00:00
class CustomEmojiManager final : public base::has_weak_ptr {
public:
using SizeTag = CustomEmojiSizeTag;
2022-06-28 13:13:20 +00:00
CustomEmojiManager(not_null<Session*> owner);
~CustomEmojiManager();
[[nodiscard]] std::unique_ptr<Ui::Text::CustomEmoji> create(
QStringView data,
2022-07-25 08:30:38 +00:00
Fn<void()> update,
SizeTag tag = SizeTag::Normal,
int sizeOverride = 0);
2022-07-25 08:30:38 +00:00
[[nodiscard]] std::unique_ptr<Ui::Text::CustomEmoji> create(
DocumentId documentId,
Fn<void()> update,
SizeTag tag = SizeTag::Normal,
int sizeOverride = 0);
2022-07-25 08:30:38 +00:00
[[nodiscard]] std::unique_ptr<Ui::Text::CustomEmoji> create(
not_null<DocumentData*> document,
Fn<void()> update,
SizeTag tag = SizeTag::Normal,
int sizeOverride = 0);
2023-01-20 16:44:08 +00:00
[[nodiscard]] Ui::Text::CustomEmojiFactory factory(
SizeTag tag = SizeTag::Normal,
int sizeOverride = 0);
class Listener {
public:
virtual void customEmojiResolveDone(
not_null<DocumentData*> document) = 0;
};
void resolve(QStringView data, not_null<Listener*> listener);
void resolve(DocumentId documentId, not_null<Listener*> listener);
void unregisterListener(not_null<Listener*> listener);
[[nodiscard]] rpl::producer<not_null<DocumentData*>> resolve(
DocumentId documentId);
[[nodiscard]] std::unique_ptr<Ui::CustomEmoji::Loader> createLoader(
not_null<DocumentData*> document,
SizeTag tag,
int sizeOverride = 0);
[[nodiscard]] std::unique_ptr<Ui::CustomEmoji::Loader> createLoader(
DocumentId documentId,
SizeTag tag,
int sizeOverride = 0);
[[nodiscard]] QString lookupSetName(uint64 setId);
[[nodiscard]] Main::Session &session() const;
[[nodiscard]] Session &owner() const;
[[nodiscard]] uint64 coloredSetId() const;
private:
2022-07-25 14:54:37 +00:00
static constexpr auto kSizeCount = int(SizeTag::kCount);
2022-06-28 13:13:20 +00:00
struct RepaintBunch {
crl::time when = 0;
std::vector<base::weak_ptr<Ui::CustomEmoji::Instance>> instances;
};
2022-08-31 08:29:09 +00:00
struct LoaderWithSetId {
std::unique_ptr<Ui::CustomEmoji::Loader> loader;
uint64 setId = 0;
bool colored = false;
2022-08-31 08:29:09 +00:00
};
[[nodiscard]] LoaderWithSetId createLoaderWithSetId(
not_null<DocumentData*> document,
SizeTag tag,
int sizeOverride = 0);
[[nodiscard]] LoaderWithSetId createLoaderWithSetId(
DocumentId documentId,
SizeTag tag,
int sizeOverride = 0);
2022-07-05 19:36:25 +00:00
void request();
void requestFinished();
2022-06-28 13:13:20 +00:00
void repaintLater(
not_null<Ui::CustomEmoji::Instance*> instance,
Ui::CustomEmoji::RepaintRequest request);
void scheduleRepaintTimer();
bool checkEmptyRepaints();
2022-06-28 13:13:20 +00:00
void invokeRepaints();
2022-08-31 08:29:09 +00:00
void fillColoredFlags(not_null<DocumentData*> document);
void processLoaders(not_null<DocumentData*> document);
void processListeners(not_null<DocumentData*> document);
void requestSetFor(not_null<DocumentData*> document);
[[nodiscard]] Ui::CustomEmoji::Preview prepareNonExactPreview(
DocumentId documentId,
SizeTag tag,
int sizeOverride) const;
2022-07-25 08:30:38 +00:00
template <typename LoaderFactory>
[[nodiscard]] std::unique_ptr<Ui::Text::CustomEmoji> create(
DocumentId documentId,
Fn<void()> update,
SizeTag tag,
int sizeOverride,
2022-07-25 08:30:38 +00:00
LoaderFactory factory);
[[nodiscard]] static int SizeIndex(SizeTag tag);
const not_null<Session*> _owner;
2022-07-25 14:54:37 +00:00
std::array<
base::flat_map<
DocumentId,
2022-07-25 14:54:37 +00:00
std::unique_ptr<Ui::CustomEmoji::Instance>>,
kSizeCount> _instances;
std::array<
base::flat_map<
DocumentId,
2022-07-25 14:54:37 +00:00
std::vector<base::weak_ptr<CustomEmojiLoader>>>,
kSizeCount> _loaders;
base::flat_map<
DocumentId,
base::flat_set<not_null<Listener*>>> _resolvers;
base::flat_map<
not_null<Listener*>,
base::flat_set<DocumentId>> _listeners;
base::flat_set<DocumentId> _pendingForRequest;
2022-08-31 08:29:09 +00:00
base::flat_map<
uint64,
base::flat_set<
not_null<Ui::CustomEmoji::Instance*>>> _coloredSetPending;
2022-07-05 19:36:25 +00:00
mtpRequestId _requestId = 0;
2022-08-31 08:29:09 +00:00
uint64 _coloredSetId = 0;
2022-06-28 13:13:20 +00:00
base::flat_map<crl::time, RepaintBunch> _repaints;
crl::time _repaintNext = 0;
base::Timer _repaintTimer;
bool _repaintTimerScheduled = false;
bool _requestSetsScheduled = false;
2022-06-28 13:13:20 +00:00
#if 0 // inject-to-on_main
crl::time _repaintsLastAdded = 0;
rpl::lifetime _repaintsLifetime;
#endif
2022-08-31 08:29:09 +00:00
rpl::lifetime _lifetime;
};
[[nodiscard]] int FrameSizeFromTag(CustomEmojiManager::SizeTag tag);
2022-11-02 12:39:13 +00:00
[[nodiscard]] QString SerializeCustomEmojiId(DocumentId id);
[[nodiscard]] QString SerializeCustomEmojiId(
not_null<DocumentData*> document);
2022-11-02 12:39:13 +00:00
[[nodiscard]] DocumentId ParseCustomEmojiData(QStringView data);
[[nodiscard]] TextWithEntities SingleCustomEmoji(DocumentId id);
[[nodiscard]] TextWithEntities SingleCustomEmoji(
not_null<DocumentData*> document);
[[nodiscard]] bool AllowEmojiWithoutPremium(not_null<PeerData*> peer);
2022-07-05 19:36:25 +00:00
void InsertCustomEmoji(
not_null<Ui::InputField*> field,
not_null<DocumentData*> document);
[[nodiscard]] Ui::Text::CustomEmojiFactory ReactedMenuFactory(
not_null<Main::Session*> session);
} // namespace Data