/* This file is part of Telegram Desktop, the official desktop version of Telegram messaging app, see https://telegram.org Telegram Desktop is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. It is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. In addition, as a special exception, the copyright holders give permission to link the code of portions of this program with the OpenSSL library. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once #include #include "base/timer.h" #include "core/single_timer.h" #include "mtproto/sender.h" #include "base/flat_map.h" #include "base/flat_set.h" #include "chat_helpers/stickers.h" class TaskQueue; class AuthSession; enum class SparseIdsLoadDirection; struct MessageGroupId; struct SendingAlbum; enum class SendMediaType; namespace Storage { enum class SharedMediaType : char; struct PreparedList; } // namespace Storage namespace Api { inline const MTPVector *getChatsFromMessagesChats(const MTPmessages_Chats &chats) { switch (chats.type()) { case mtpc_messages_chats: return &chats.c_messages_chats().vchats; case mtpc_messages_chatsSlice: return &chats.c_messages_chatsSlice().vchats; } return nullptr; } } // namespace Api class ApiWrap : private MTP::Sender, private base::Subscriber { public: ApiWrap(not_null session); void start(); void applyUpdates(const MTPUpdates &updates, uint64 sentMessageRandomId = 0); using RequestMessageDataCallback = base::lambda; void requestMessageData(ChannelData *channel, MsgId msgId, RequestMessageDataCallback callback); void requestFullPeer(PeerData *peer); void requestPeer(PeerData *peer); void requestPeers(const QList &peers); void requestLastParticipants(not_null channel); void requestBots(not_null channel); void requestAdmins(not_null channel); void requestParticipantsCountDelayed(not_null channel); void requestChannelMembersForAdd( not_null channel, base::lambda callback); void processFullPeer(PeerData *peer, const MTPmessages_ChatFull &result); void processFullPeer(UserData *user, const MTPUserFull &result); void requestSelfParticipant(ChannelData *channel); void kickParticipant(not_null chat, not_null user); void kickParticipant( not_null channel, not_null user, const MTPChannelBannedRights ¤tRights); void unblockParticipant( not_null channel, not_null user); void requestWebPageDelayed(WebPageData *page); void clearWebPageRequest(WebPageData *page); void clearWebPageRequests(); void scheduleStickerSetRequest(uint64 setId, uint64 access); void requestStickerSets(); void saveStickerSets(const Stickers::Order &localOrder, const Stickers::Order &localRemoved); void updateStickers(); void setGroupStickerSet(not_null megagroup, const MTPInputStickerSet &set); void joinChannel(ChannelData *channel); void leaveChannel(ChannelData *channel); void blockUser(UserData *user); void unblockUser(UserData *user); void exportInviteLink(PeerData *peer); void requestNotifySetting(PeerData *peer); void saveDraftToCloudDelayed(History *history); void savePrivacy(const MTPInputPrivacyKey &key, QVector &&rules); void handlePrivacyChange(mtpTypeId keyTypeId, const MTPVector &rules); int onlineTillFromStatus(const MTPUserStatus &status, int currentOnlineTill); void clearHistory(not_null peer); base::Observable &fullPeerUpdated() { return _fullPeerUpdated; } bool isQuitPrevent(); void applyUpdatesNoPtsCheck(const MTPUpdates &updates); void applyUpdateNoPtsCheck(const MTPUpdate &update); void jumpToDate(not_null peer, const QDate &date); void preloadEnoughUnreadMentions(not_null history); void checkForUnreadMentions(const base::flat_set &possiblyReadMentions, ChannelData *channel = nullptr); void editChatAdmins( not_null chat, bool adminsEnabled, base::flat_set> &&admins); using SliceType = SparseIdsLoadDirection; void requestSharedMedia( not_null peer, Storage::SharedMediaType type, MsgId messageId, SliceType slice); void requestSharedMediaCount( not_null peer, Storage::SharedMediaType type); void requestUserPhotos( not_null user, PhotoId afterId); void stickerSetInstalled(uint64 setId) { _stickerSetInstalled.fire_copy(setId); } auto stickerSetInstalled() const { return _stickerSetInstalled.events(); } void readFeaturedSetDelayed(uint64 setId); void parseChannelParticipants( not_null channel, const MTPchannels_ChannelParticipants &result, base::lambda &list)> callbackList, base::lambda callbackNotModified = nullptr); void parseRecentChannelParticipants( not_null channel, const MTPchannels_ChannelParticipants &result, base::lambda &list)> callbackList, base::lambda callbackNotModified = nullptr); struct SendOptions { SendOptions(not_null history) : history(history) { } not_null history; MsgId replyTo = 0; WebPageId webPageId = 0; bool clearDraft = false; bool generateLocal = true; }; rpl::producer sendActions() const { return _sendActions.events(); } void sendAction(const SendOptions &options); void forwardMessages( HistoryItemsList &&items, const SendOptions &options, base::lambda_once &&successCallback = nullptr); void shareContact( const QString &phone, const QString &firstName, const QString &lastName, const SendOptions &options); void shareContact(not_null user, const SendOptions &options); void readServerHistory(not_null history); void readServerHistoryForce(not_null history); void sendVoiceMessage( QByteArray result, VoiceWaveform waveform, int duration, const SendOptions &options); void sendFiles( Storage::PreparedList &&list, SendMediaType type, QString caption, std::shared_ptr album, const SendOptions &options); void sendFile( const QByteArray &fileContent, SendMediaType type, const SendOptions &options); void sendUploadedPhoto( FullMsgId localId, const MTPInputFile &file, bool silent); void sendUploadedDocument( FullMsgId localId, const MTPInputFile &file, const base::optional &thumb, bool silent); void cancelLocalItem(not_null item); ~ApiWrap(); private: struct MessageDataRequest { using Callbacks = QList; mtpRequestId requestId = 0; Callbacks callbacks; }; using MessageDataRequests = QMap; using SharedMediaType = Storage::SharedMediaType; void requestAppChangelogs(); void addLocalChangelogs(int oldAppVersion); void updatesReceived(const MTPUpdates &updates); void checkQuitPreventFinished(); void saveDraftsToCloud(); void resolveMessageDatas(); void gotMessageDatas(ChannelData *channel, const MTPmessages_Messages &result, mtpRequestId requestId); void finalizeMessageDataRequest( ChannelData *channel, mtpRequestId requestId); QVector collectMessageIds(const MessageDataRequests &requests); MessageDataRequests *messageDataRequests(ChannelData *channel, bool onlyExisting = false); void gotChatFull(PeerData *peer, const MTPmessages_ChatFull &result, mtpRequestId req); void gotUserFull(UserData *user, const MTPUserFull &result, mtpRequestId req); void applyLastParticipantsList( not_null channel, int availableCount, const QVector &list); void applyBotsList( not_null channel, int availableCount, const QVector &list); void applyAdminsList( not_null channel, int availableCount, const QVector &list); void resolveWebPages(); void gotWebPages(ChannelData *channel, const MTPmessages_Messages &result, mtpRequestId req); void gotStickerSet(uint64 setId, const MTPmessages_StickerSet &result); PeerData *notifySettingReceived(MTPInputNotifyPeer peer, const MTPPeerNotifySettings &settings); void stickerSetDisenabled(mtpRequestId requestId); void stickersSaveOrder(); void requestStickers(TimeId now); void requestRecentStickers(TimeId now); void requestFavedStickers(TimeId now); void requestFeaturedStickers(TimeId now); void requestSavedGifs(TimeId now); void readFeaturedSets(); void cancelEditChatAdmins(not_null chat); void saveChatAdmins(not_null chat); void sendSaveChatAdminsRequests(not_null chat); void refreshChannelAdmins( not_null channel, const QVector &participants); template void requestMessageAfterDate( not_null peer, const QDate &date, Callback &&callback); void sharedMediaDone( not_null peer, SharedMediaType type, MsgId messageId, SliceType slice, const MTPmessages_Messages &result); void userPhotosDone( not_null user, PhotoId photoId, const MTPphotos_Photos &result); void sendSharedContact( const QString &phone, const QString &firstName, const QString &lastName, UserId userId, const SendOptions &options); void sendReadRequest(not_null peer, MsgId upTo); int applyAffectedHistory( not_null peer, const MTPmessages_AffectedHistory &result); void applyAffectedMessages(const MTPmessages_AffectedMessages &result); void applyAffectedMessages( not_null peer, const MTPmessages_AffectedMessages &result); void sendMessageFail(const RPCError &error); void uploadAlbumMedia( not_null item, const MessageGroupId &groupId, const MTPInputMedia &media); void sendAlbumWithUploaded( not_null item, const MessageGroupId &groupId, const MTPInputMedia &media); void sendAlbumWithCancelled( not_null item, const MessageGroupId &groupId); void sendAlbumIfReady(not_null album); void sendMedia( not_null item, const MTPInputMedia &media, bool silent); void sendMediaWithRandomId( not_null item, const MTPInputMedia &media, bool silent, uint64 randomId); not_null _session; mtpRequestId _changelogSubscription = 0; MessageDataRequests _messageDataRequests; QMap _channelMessageDataRequests; SingleQueuedInvokation _messageDataResolveDelayed; using PeerRequests = QMap; PeerRequests _fullPeerRequests; PeerRequests _peerRequests; PeerRequests _participantsRequests; PeerRequests _botsRequests; PeerRequests _adminsRequests; base::DelayedCallTimer _participantsCountRequestTimer; ChannelData *_channelMembersForAdd = nullptr; mtpRequestId _channelMembersForAddRequestId = 0; base::lambda _channelMembersForAddCallback; using KickRequest = std::pair< not_null, not_null>; base::flat_map _kickRequests; QMap _selfParticipantRequests; QMap _webPagesPending; base::Timer _webPagesTimer; QMap > _stickerSetRequests; QMap _channelAmInRequests; QMap _blockRequests; QMap _exportInviteRequests; QMap _notifySettingRequests; QMap _draftsSaveRequestIds; base::Timer _draftsSaveTimer; base::flat_set _stickerSetDisenableRequests; Stickers::Order _stickersOrder; mtpRequestId _stickersReorderRequestId = 0; mtpRequestId _stickersClearRecentRequestId = 0; mtpRequestId _stickersUpdateRequest = 0; mtpRequestId _recentStickersUpdateRequest = 0; mtpRequestId _favedStickersUpdateRequest = 0; mtpRequestId _featuredStickersUpdateRequest = 0; mtpRequestId _savedGifsUpdateRequest = 0; base::Timer _featuredSetsReadTimer; base::flat_set _featuredSetsRead; QMap _privacySaveRequests; mtpRequestId _contactsStatusesRequestId = 0; base::flat_map, mtpRequestId> _unreadMentionsRequests; base::flat_map< not_null, mtpRequestId> _chatAdminsEnabledRequests; base::flat_map< not_null, base::flat_set>> _chatAdminsToSave; base::flat_map< not_null, base::flat_set> _chatAdminsSaveRequests; base::flat_map, SharedMediaType, MsgId, SliceType>, mtpRequestId> _sharedMediaRequests; base::flat_map, mtpRequestId> _userPhotosRequests; rpl::event_stream _sendActions; struct ReadRequest { ReadRequest(mtpRequestId requestId, MsgId upTo) : requestId(requestId) , upTo(upTo) { } mtpRequestId requestId = 0; MsgId upTo = 0; }; base::flat_map, ReadRequest> _readRequests; base::flat_map, MsgId> _readRequestsPending; std::unique_ptr _fileLoader; base::flat_map> _sendingAlbums; base::Observable _fullPeerUpdated; rpl::event_stream _stickerSetInstalled; };