/* 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 "base/flags.h" class History; class PeerData; class HistoryItem; namespace Dialogs { class Entry; } // namespace Dialogs namespace Data { namespace details { template inline constexpr int CountBit(Flag Last = Flag::LastUsedBit) { auto i = 0; while ((1ULL << i) != static_cast(Last)) { ++i; Assert(i != 64); } return (i + 1); } } // namespace details struct NameUpdate { NameUpdate( not_null peer, base::flat_set oldFirstLetters) : peer(peer) , oldFirstLetters(std::move(oldFirstLetters)) { } not_null peer; base::flat_set oldFirstLetters; }; struct PeerUpdate { enum class Flag : uint32 { None = 0, // Common flags Name = (1 << 0), Username = (1 << 1), Photo = (1 << 2), About = (1 << 3), Notifications = (1 << 4), Migration = (1 << 5), UnavailableReason = (1 << 6), PinnedMessage = (1 << 7), // For users CanShareContact = (1 << 8), IsContact = (1 << 9), PhoneNumber = (1 << 10), IsBlocked = (1 << 11), OnlineStatus = (1 << 12), BotCommands = (1 << 13), BotCanBeInvited = (1 << 14), CommonChats = (1 << 15), HasCalls = (1 << 16), SupportInfo = (1 << 17), IsBot = (1 << 18), // For chats and channels InviteLink = (1 << 19), Members = (1 << 20), Admins = (1 << 21), BannedUsers = (1 << 22), Rights = (1 << 23), // For channels ChannelAmIn = (1 << 24), StickersSet = (1 << 25), ChannelLinkedChat = (1 << 26), ChannelLocation = (1 << 27), Slowmode = (1 << 28), // For iteration LastUsedBit = (1 << 28), }; using Flags = base::flags; friend inline constexpr auto is_flag_type(Flag) { return true; } not_null peer; Flags flags = 0; }; struct HistoryUpdate { enum class Flag : uint32 { None = 0, IsPinned = (1 << 0), UnreadView = (1 << 1), TopPromoted = (1 << 2), Folder = (1 << 3), UnreadMentions = (1 << 4), LocalMessages = (1 << 5), ChatOccupied = (1 << 6), MessageSent = (1 << 7), ForwardDraft = (1 << 8), OutboxRead = (1 << 9), BotKeyboard = (1 << 10), CloudDraft = (1 << 11), LastUsedBit = (1 << 11), }; using Flags = base::flags; friend inline constexpr auto is_flag_type(Flag) { return true; } not_null history; Flags flags = 0; }; struct MessageUpdate { enum class Flag : uint32 { None = 0, Edited = (1 << 0), Destroyed = (1 << 1), DialogRowRepaint = (1 << 2), DialogRowRefresh = (1 << 3), CallAdded = (1 << 4), ReplyMarkup = (1 << 5), BotCallbackSent = (1 << 6), LastUsedBit = (1 << 6), }; using Flags = base::flags; friend inline constexpr auto is_flag_type(Flag) { return true; } not_null item; Flags flags = 0; }; struct EntryUpdate { enum class Flag : uint32 { None = 0, Repaint = (1 << 0), LastUsedBit = (1 << 0), }; using Flags = base::flags; friend inline constexpr auto is_flag_type(Flag) { return true; } not_null entry; Flags flags = 0; }; class Changes final { public: explicit Changes(not_null session); [[nodiscard]] Main::Session &session() const; void nameUpdated( not_null peer, base::flat_set oldFirstLetters); [[nodiscard]] rpl::producer realtimeNameUpdates() const; [[nodiscard]] rpl::producer realtimeNameUpdates( not_null peer) const; void peerUpdated(not_null peer, PeerUpdate::Flags flags); [[nodiscard]] rpl::producer peerUpdates( PeerUpdate::Flags flags) const; [[nodiscard]] rpl::producer peerUpdates( not_null peer, PeerUpdate::Flags flags) const; [[nodiscard]] rpl::producer peerFlagsValue( not_null peer, PeerUpdate::Flags flags) const; [[nodiscard]] rpl::producer realtimePeerUpdates( PeerUpdate::Flag flag) const; void historyUpdated( not_null history, HistoryUpdate::Flags flags); [[nodiscard]] rpl::producer historyUpdates( HistoryUpdate::Flags flags) const; [[nodiscard]] rpl::producer historyUpdates( not_null history, HistoryUpdate::Flags flags) const; [[nodiscard]] rpl::producer historyFlagsValue( not_null history, HistoryUpdate::Flags flags) const; [[nodiscard]] rpl::producer realtimeHistoryUpdates( HistoryUpdate::Flag flag) const; void messageUpdated( not_null item, MessageUpdate::Flags flags); [[nodiscard]] rpl::producer messageUpdates( MessageUpdate::Flags flags) const; [[nodiscard]] rpl::producer messageUpdates( not_null item, MessageUpdate::Flags flags) const; [[nodiscard]] rpl::producer messageFlagsValue( not_null item, MessageUpdate::Flags flags) const; [[nodiscard]] rpl::producer realtimeMessageUpdates( MessageUpdate::Flag flag) const; void entryUpdated( not_null entry, EntryUpdate::Flags flags); [[nodiscard]] rpl::producer entryUpdates( EntryUpdate::Flags flags) const; [[nodiscard]] rpl::producer entryUpdates( not_null entry, EntryUpdate::Flags flags) const; [[nodiscard]] rpl::producer entryFlagsValue( not_null entry, EntryUpdate::Flags flags) const; [[nodiscard]] rpl::producer realtimeEntryUpdates( EntryUpdate::Flag flag) const; void sendNotifications(); private: template class Manager final { public: using Flag = typename UpdateType::Flag; using Flags = typename UpdateType::Flags; void updated( not_null data, Flags flags, bool dropScheduled = false); [[nodiscard]] rpl::producer updates(Flags flags) const; [[nodiscard]] rpl::producer updates( not_null data, Flags flags) const; [[nodiscard]] rpl::producer flagsValue( not_null data, Flags flags) const; [[nodiscard]] rpl::producer realtimeUpdates( Flag flag) const; void sendNotifications(); private: static constexpr auto kCount = details::CountBit(); void sendRealtimeNotifications(not_null data, Flags flags); std::array, kCount> _realtimeStreams; base::flat_map, Flags> _updates; rpl::event_stream _stream; }; void scheduleNotifications(); const not_null _session; rpl::event_stream _nameStream; Manager _peerChanges; Manager _historyChanges; Manager _messageChanges; Manager _entryChanges; bool _notify = false; }; } // namespace Data