/* 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 Main { class Session; } // namespace Main 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 = (1U << 0), Username = (1U << 1), Photo = (1U << 2), About = (1U << 3), Notifications = (1U << 4), Migration = (1U << 5), UnavailableReason = (1U << 6), ChatThemeEmoji = (1U << 7), IsBlocked = (1U << 8), MessagesTTL = (1U << 9), // For users CanShareContact = (1U << 10), IsContact = (1U << 11), PhoneNumber = (1U << 12), OnlineStatus = (1U << 13), BotCommands = (1U << 14), BotCanBeInvited = (1U << 15), BotStartToken = (1U << 16), CommonChats = (1U << 17), HasCalls = (1U << 18), SupportInfo = (1U << 19), IsBot = (1U << 20), // For chats and channels InviteLinks = (1U << 21), Members = (1U << 22), Admins = (1U << 23), BannedUsers = (1U << 24), Rights = (1U << 25), // For channels ChannelAmIn = (1U << 26), StickersSet = (1U << 27), ChannelLinkedChat = (1U << 28), ChannelLocation = (1U << 29), Slowmode = (1U << 30), GroupCall = (1U << 31), // For iteration LastUsedBit = (1U << 31), }; 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 = (1U << 0), UnreadView = (1U << 1), TopPromoted = (1U << 2), Folder = (1U << 3), UnreadMentions = (1U << 4), LocalMessages = (1U << 5), ChatOccupied = (1U << 6), MessageSent = (1U << 7), ScheduledSent = (1U << 8), ForwardDraft = (1U << 9), OutboxRead = (1U << 10), BotKeyboard = (1U << 11), CloudDraft = (1U << 12), LocalDraftSet = (1U << 13), PinnedMessages = (1U << 14), LastUsedBit = (1U << 14), }; 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 = (1U << 0), Destroyed = (1U << 1), DialogRowRepaint = (1U << 2), DialogRowRefresh = (1U << 3), NewAdded = (1U << 4), ReplyMarkup = (1U << 5), BotCallbackSent = (1U << 6), NewMaybeAdded = (1U << 7), RepliesUnreadCount = (1U << 8), LastUsedBit = (1U << 8), }; 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 = (1U << 0), LastUsedBit = (1U << 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