489 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			489 lines
		
	
	
	
		
			13 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
 | 
						|
*/
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include "data/data_peer.h"
 | 
						|
#include "data/data_pts_waiter.h"
 | 
						|
#include "data/data_location.h"
 | 
						|
#include "data/data_chat_participant_status.h"
 | 
						|
 | 
						|
struct ChannelLocation {
 | 
						|
	QString address;
 | 
						|
	Data::LocationPoint point;
 | 
						|
 | 
						|
	friend inline bool operator==(
 | 
						|
			const ChannelLocation &a,
 | 
						|
			const ChannelLocation &b) {
 | 
						|
		return a.address.isEmpty()
 | 
						|
			? b.address.isEmpty()
 | 
						|
			: (a.address == b.address && a.point == b.point);
 | 
						|
	}
 | 
						|
	friend inline bool operator!=(
 | 
						|
			const ChannelLocation &a,
 | 
						|
			const ChannelLocation &b) {
 | 
						|
		return !(a == b);
 | 
						|
	}
 | 
						|
};
 | 
						|
 | 
						|
enum class ChannelDataFlag {
 | 
						|
	Left = (1 << 0),
 | 
						|
	Creator = (1 << 1),
 | 
						|
	Forbidden = (1 << 2),
 | 
						|
	CallActive = (1 << 3),
 | 
						|
	CallNotEmpty = (1 << 4),
 | 
						|
	Signatures = (1 << 5),
 | 
						|
	Verified = (1 << 6),
 | 
						|
	Scam = (1 << 7),
 | 
						|
	Fake = (1 << 8),
 | 
						|
	Megagroup = (1 << 9),
 | 
						|
	Broadcast = (1 << 10),
 | 
						|
	Gigagroup = (1 << 11),
 | 
						|
	Username = (1 << 12),
 | 
						|
	Location = (1 << 13),
 | 
						|
	CanSetUsername = (1 << 14),
 | 
						|
	CanSetStickers = (1 << 15),
 | 
						|
	PreHistoryHidden = (1 << 16),
 | 
						|
	CanViewParticipants = (1 << 17),
 | 
						|
	HasLink = (1 << 18),
 | 
						|
	SlowmodeEnabled = (1 << 19),
 | 
						|
	NoForwards = (1 << 20),
 | 
						|
};
 | 
						|
inline constexpr bool is_flag_type(ChannelDataFlag) { return true; };
 | 
						|
using ChannelDataFlags = base::flags<ChannelDataFlag>;
 | 
						|
 | 
						|
class MegagroupInfo {
 | 
						|
public:
 | 
						|
	struct Admin {
 | 
						|
		explicit Admin(ChatAdminRightsInfo rights)
 | 
						|
		: rights(rights) {
 | 
						|
		}
 | 
						|
		Admin(ChatAdminRightsInfo rights, bool canEdit)
 | 
						|
		: rights(rights)
 | 
						|
		, canEdit(canEdit) {
 | 
						|
		}
 | 
						|
		ChatAdminRightsInfo rights;
 | 
						|
		bool canEdit = false;
 | 
						|
	};
 | 
						|
 | 
						|
	struct Restricted {
 | 
						|
		explicit Restricted(ChatRestrictionsInfo rights)
 | 
						|
		: rights(rights) {
 | 
						|
		}
 | 
						|
		ChatRestrictionsInfo rights;
 | 
						|
	};
 | 
						|
 | 
						|
	ChatData *getMigrateFromChat() const;
 | 
						|
	void setMigrateFromChat(ChatData *chat);
 | 
						|
 | 
						|
	const ChannelLocation *getLocation() const;
 | 
						|
	void setLocation(const ChannelLocation &location);
 | 
						|
 | 
						|
	bool updateBotCommands(const MTPVector<MTPBotInfo> &data);
 | 
						|
	bool updateBotCommands(
 | 
						|
		UserId botId,
 | 
						|
		const MTPVector<MTPBotCommand> &data);
 | 
						|
	[[nodiscard]] auto botCommands() const
 | 
						|
		-> const base::flat_map<UserId, std::vector<BotCommand>> & {
 | 
						|
		return _botCommands;
 | 
						|
	}
 | 
						|
 | 
						|
	std::deque<not_null<UserData*>> lastParticipants;
 | 
						|
	base::flat_map<not_null<UserData*>, Admin> lastAdmins;
 | 
						|
	base::flat_map<not_null<UserData*>, Restricted> lastRestricted;
 | 
						|
	base::flat_set<not_null<PeerData*>> markupSenders;
 | 
						|
	base::flat_set<not_null<UserData*>> bots;
 | 
						|
 | 
						|
	// For admin badges, full admins list with ranks.
 | 
						|
	base::flat_map<UserId, QString> admins;
 | 
						|
 | 
						|
	UserData *creator = nullptr; // nullptr means unknown
 | 
						|
	QString creatorRank;
 | 
						|
	int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
 | 
						|
	bool joinedMessageFound = false;
 | 
						|
	StickerSetIdentifier stickerSet;
 | 
						|
 | 
						|
	enum LastParticipantsStatus {
 | 
						|
		LastParticipantsUpToDate       = 0x00,
 | 
						|
		LastParticipantsOnceReceived   = 0x01,
 | 
						|
		LastParticipantsCountOutdated  = 0x02,
 | 
						|
	};
 | 
						|
	mutable int lastParticipantsStatus = LastParticipantsUpToDate;
 | 
						|
	int lastParticipantsCount = 0;
 | 
						|
 | 
						|
private:
 | 
						|
	ChatData *_migratedFrom = nullptr;
 | 
						|
	ChannelLocation _location;
 | 
						|
	base::flat_map<UserId, std::vector<BotCommand>> _botCommands;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
class ChannelData : public PeerData {
 | 
						|
public:
 | 
						|
	using Flag = ChannelDataFlag;
 | 
						|
	using Flags = Data::Flags<ChannelDataFlags>;
 | 
						|
 | 
						|
	using AdminRight = ChatAdminRight;
 | 
						|
	using Restriction = ChatRestriction;
 | 
						|
	using AdminRights = ChatAdminRights;
 | 
						|
	using Restrictions = ChatRestrictions;
 | 
						|
	using AdminRightFlags = Data::Flags<AdminRights>;
 | 
						|
	using RestrictionFlags = Data::Flags<Restrictions>;
 | 
						|
 | 
						|
	ChannelData(not_null<Data::Session*> owner, PeerId id);
 | 
						|
 | 
						|
	void setName(const QString &name, const QString &username);
 | 
						|
	void setPhoto(const MTPChatPhoto &photo);
 | 
						|
	void setAccessHash(uint64 accessHash);
 | 
						|
 | 
						|
	void setFlags(ChannelDataFlags which) {
 | 
						|
		_flags.set(which);
 | 
						|
	}
 | 
						|
	void addFlags(ChannelDataFlags which) {
 | 
						|
		_flags.add(which);
 | 
						|
	}
 | 
						|
	void removeFlags(ChannelDataFlags which) {
 | 
						|
		_flags.remove(which);
 | 
						|
	}
 | 
						|
	[[nodiscard]] auto flags() const {
 | 
						|
		return _flags.current();
 | 
						|
	}
 | 
						|
	[[nodiscard]] auto flagsValue() const {
 | 
						|
		return _flags.value();
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] int membersCount() const {
 | 
						|
		return std::max(_membersCount, 1);
 | 
						|
	}
 | 
						|
	void setMembersCount(int newMembersCount);
 | 
						|
	[[nodiscard]] bool membersCountKnown() const {
 | 
						|
		return (_membersCount >= 0);
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] int adminsCount() const {
 | 
						|
		return _adminsCount;
 | 
						|
	}
 | 
						|
	void setAdminsCount(int newAdminsCount);
 | 
						|
 | 
						|
	[[nodiscard]] int restrictedCount() const {
 | 
						|
		return _restrictedCount;
 | 
						|
	}
 | 
						|
	void setRestrictedCount(int newRestrictedCount);
 | 
						|
 | 
						|
	[[nodiscard]] int kickedCount() const {
 | 
						|
		return _kickedCount;
 | 
						|
	}
 | 
						|
	void setKickedCount(int newKickedCount);
 | 
						|
 | 
						|
	[[nodiscard]] int pendingRequestsCount() const {
 | 
						|
		return _pendingRequestsCount;
 | 
						|
	}
 | 
						|
	[[nodiscard]] const std::vector<UserId> &recentRequesters() const {
 | 
						|
		return _recentRequesters;
 | 
						|
	}
 | 
						|
	void setPendingRequestsCount(
 | 
						|
		int count,
 | 
						|
		const QVector<MTPlong> &recentRequesters);
 | 
						|
	void setPendingRequestsCount(
 | 
						|
		int count,
 | 
						|
		std::vector<UserId> recentRequesters);
 | 
						|
 | 
						|
	[[nodiscard]] bool haveLeft() const {
 | 
						|
		return flags() & Flag::Left;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool amIn() const {
 | 
						|
		return !isForbidden() && !haveLeft();
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool addsSignature() const {
 | 
						|
		return flags() & Flag::Signatures;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isForbidden() const {
 | 
						|
		return flags() & Flag::Forbidden;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isVerified() const {
 | 
						|
		return flags() & Flag::Verified;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isScam() const {
 | 
						|
		return flags() & Flag::Scam;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isFake() const {
 | 
						|
		return flags() & Flag::Fake;
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] static ChatRestrictionsInfo KickedRestrictedRights(
 | 
						|
		not_null<PeerData*> participant);
 | 
						|
	static constexpr auto kRestrictUntilForever = TimeId(INT_MAX);
 | 
						|
	[[nodiscard]] static bool IsRestrictedForever(TimeId until) {
 | 
						|
		return !until || (until == kRestrictUntilForever);
 | 
						|
	}
 | 
						|
	void applyEditAdmin(
 | 
						|
		not_null<UserData*> user,
 | 
						|
		ChatAdminRightsInfo oldRights,
 | 
						|
		ChatAdminRightsInfo newRights,
 | 
						|
		const QString &rank);
 | 
						|
	void applyEditBanned(
 | 
						|
		not_null<PeerData*> participant,
 | 
						|
		ChatRestrictionsInfo oldRights,
 | 
						|
		ChatRestrictionsInfo newRights);
 | 
						|
 | 
						|
	void markForbidden();
 | 
						|
 | 
						|
	[[nodiscard]] bool isGroupAdmin(not_null<UserData*> user) const;
 | 
						|
	[[nodiscard]] bool lastParticipantsRequestNeeded() const;
 | 
						|
	[[nodiscard]] bool isMegagroup() const {
 | 
						|
		return flags() & Flag::Megagroup;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isBroadcast() const {
 | 
						|
		return flags() & Flag::Broadcast;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isGigagroup() const {
 | 
						|
		return flags() & Flag::Gigagroup;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool hasUsername() const {
 | 
						|
		return flags() & Flag::Username;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool hasLocation() const {
 | 
						|
		return flags() & Flag::Location;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool isPublic() const {
 | 
						|
		return hasUsername() || hasLocation();
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool amCreator() const {
 | 
						|
		return flags() & Flag::Creator;
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] auto adminRights() const {
 | 
						|
		return _adminRights.current();
 | 
						|
	}
 | 
						|
	[[nodiscard]] auto adminRightsValue() const {
 | 
						|
		return _adminRights.value();
 | 
						|
	}
 | 
						|
	void setAdminRights(ChatAdminRights rights);
 | 
						|
	[[nodiscard]] bool hasAdminRights() const {
 | 
						|
		return (adminRights() != 0);
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] auto restrictions() const {
 | 
						|
		return _restrictions.current();
 | 
						|
	}
 | 
						|
	[[nodiscard]] auto restrictionsValue() const {
 | 
						|
		return _restrictions.value();
 | 
						|
	}
 | 
						|
	[[nodiscard]] TimeId restrictedUntil() const {
 | 
						|
		return _restrictedUntil;
 | 
						|
	}
 | 
						|
	void setRestrictions(ChatRestrictionsInfo rights);
 | 
						|
	[[nodiscard]] bool hasRestrictions() const {
 | 
						|
		return (restrictions() != 0);
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool hasRestrictions(TimeId now) const {
 | 
						|
		return hasRestrictions()
 | 
						|
			&& (restrictedUntil() > now);
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] auto defaultRestrictions() const {
 | 
						|
		return _defaultRestrictions.current();
 | 
						|
	}
 | 
						|
	[[nodiscard]] auto defaultRestrictionsValue() const {
 | 
						|
		return _defaultRestrictions.value();
 | 
						|
	}
 | 
						|
	void setDefaultRestrictions(ChatRestrictions rights);
 | 
						|
 | 
						|
	// Like in ChatData.
 | 
						|
	[[nodiscard]] bool canWrite() const;
 | 
						|
	[[nodiscard]] bool allowsForwarding() const;
 | 
						|
	[[nodiscard]] bool canEditInformation() const;
 | 
						|
	[[nodiscard]] bool canEditPermissions() const;
 | 
						|
	[[nodiscard]] bool canEditUsername() const;
 | 
						|
	[[nodiscard]] bool canEditPreHistoryHidden() const;
 | 
						|
	[[nodiscard]] bool canAddMembers() const;
 | 
						|
	[[nodiscard]] bool canAddAdmins() const;
 | 
						|
	[[nodiscard]] bool canBanMembers() const;
 | 
						|
	[[nodiscard]] bool canSendPolls() const;
 | 
						|
	[[nodiscard]] bool anyoneCanAddMembers() const;
 | 
						|
 | 
						|
	[[nodiscard]] bool canEditMessages() const;
 | 
						|
	[[nodiscard]] bool canDeleteMessages() const;
 | 
						|
	[[nodiscard]] bool hiddenPreHistory() const;
 | 
						|
	[[nodiscard]] bool canPublish() const;
 | 
						|
	[[nodiscard]] bool canViewMembers() const;
 | 
						|
	[[nodiscard]] bool canViewAdmins() const;
 | 
						|
	[[nodiscard]] bool canViewBanned() const;
 | 
						|
	[[nodiscard]] bool canEditSignatures() const;
 | 
						|
	[[nodiscard]] bool canEditStickers() const;
 | 
						|
	[[nodiscard]] bool canDelete() const;
 | 
						|
	[[nodiscard]] bool canEditAdmin(not_null<UserData*> user) const;
 | 
						|
	[[nodiscard]] bool canRestrictParticipant(
 | 
						|
		not_null<PeerData*> participant) const;
 | 
						|
 | 
						|
	void setInviteLink(const QString &newInviteLink);
 | 
						|
	[[nodiscard]] QString inviteLink() const {
 | 
						|
		return _inviteLink;
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool canHaveInviteLink() const;
 | 
						|
 | 
						|
	void setLocation(const MTPChannelLocation &data);
 | 
						|
	[[nodiscard]] const ChannelLocation *getLocation() const;
 | 
						|
 | 
						|
	void setLinkedChat(ChannelData *linked);
 | 
						|
	[[nodiscard]] ChannelData *linkedChat() const;
 | 
						|
	[[nodiscard]] bool linkedChatKnown() const;
 | 
						|
 | 
						|
	void ptsInit(int32 pts) {
 | 
						|
		_ptsWaiter.init(pts);
 | 
						|
	}
 | 
						|
	void ptsReceived(int32 pts) {
 | 
						|
		_ptsWaiter.updateAndApply(this, pts, 0);
 | 
						|
	}
 | 
						|
	bool ptsUpdateAndApply(int32 pts, int32 count) {
 | 
						|
		return _ptsWaiter.updateAndApply(this, pts, count);
 | 
						|
	}
 | 
						|
	bool ptsUpdateAndApply(
 | 
						|
		int32 pts,
 | 
						|
		int32 count,
 | 
						|
		const MTPUpdate &update) {
 | 
						|
		return _ptsWaiter.updateAndApply(this, pts, count, update);
 | 
						|
	}
 | 
						|
	bool ptsUpdateAndApply(
 | 
						|
			int32 pts,
 | 
						|
			int32 count,
 | 
						|
			const MTPUpdates &updates) {
 | 
						|
		return _ptsWaiter.updateAndApply(this, pts, count, updates);
 | 
						|
	}
 | 
						|
	[[nodiscard]] int32 pts() const {
 | 
						|
		return _ptsWaiter.current();
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool ptsInited() const {
 | 
						|
		return _ptsWaiter.inited();
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool ptsRequesting() const {
 | 
						|
		return _ptsWaiter.requesting();
 | 
						|
	}
 | 
						|
	void ptsSetRequesting(bool isRequesting) {
 | 
						|
		return _ptsWaiter.setRequesting(isRequesting);
 | 
						|
	}
 | 
						|
	// < 0 - not waiting
 | 
						|
	void ptsWaitingForShortPoll(int32 ms) {
 | 
						|
		return _ptsWaiter.setWaitingForShortPoll(this, ms);
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool ptsWaitingForSkipped() const {
 | 
						|
		return _ptsWaiter.waitingForSkipped();
 | 
						|
	}
 | 
						|
	[[nodiscard]] bool ptsWaitingForShortPoll() const {
 | 
						|
		return _ptsWaiter.waitingForShortPoll();
 | 
						|
	}
 | 
						|
 | 
						|
	void setUnavailableReasons(
 | 
						|
		std::vector<Data::UnavailableReason> &&reason);
 | 
						|
 | 
						|
	[[nodiscard]] MsgId availableMinId() const {
 | 
						|
		return _availableMinId;
 | 
						|
	}
 | 
						|
	void setAvailableMinId(MsgId availableMinId);
 | 
						|
 | 
						|
	[[nodiscard]] ChatData *getMigrateFromChat() const;
 | 
						|
	void setMigrateFromChat(ChatData *chat);
 | 
						|
 | 
						|
	[[nodiscard]] int slowmodeSeconds() const;
 | 
						|
	void setSlowmodeSeconds(int seconds);
 | 
						|
	[[nodiscard]] TimeId slowmodeLastMessage() const;
 | 
						|
	void growSlowmodeLastMessage(TimeId when);
 | 
						|
 | 
						|
	void setInvitePeek(const QString &hash, TimeId expires);
 | 
						|
	void clearInvitePeek();
 | 
						|
	[[nodiscard]] TimeId invitePeekExpires() const;
 | 
						|
	[[nodiscard]] QString invitePeekHash() const;
 | 
						|
	void privateErrorReceived();
 | 
						|
 | 
						|
	[[nodiscard]] Data::GroupCall *groupCall() const {
 | 
						|
		return _call.get();
 | 
						|
	}
 | 
						|
	void migrateCall(std::unique_ptr<Data::GroupCall> call);
 | 
						|
	void setGroupCall(
 | 
						|
		const MTPInputGroupCall &call,
 | 
						|
		TimeId scheduleDate = 0);
 | 
						|
	void clearGroupCall();
 | 
						|
	void setGroupCallDefaultJoinAs(PeerId peerId);
 | 
						|
	[[nodiscard]] PeerId groupCallDefaultJoinAs() const;
 | 
						|
 | 
						|
	void setAllowedReactions(std::vector<QString> list);
 | 
						|
	[[nodiscard]] const std::vector<QString> &allowedReactions() const;
 | 
						|
 | 
						|
	// Still public data members.
 | 
						|
	uint64 access = 0;
 | 
						|
 | 
						|
	MTPinputChannel inputChannel = MTP_inputChannelEmpty();
 | 
						|
 | 
						|
	QString username;
 | 
						|
 | 
						|
	int32 date = 0;
 | 
						|
	std::unique_ptr<MegagroupInfo> mgInfo;
 | 
						|
 | 
						|
	// > 0 - user who invited me to channel, < 0 - not in channel.
 | 
						|
	UserId inviter = 0;
 | 
						|
	TimeId inviteDate = 0;
 | 
						|
	bool inviteViaRequest = false;
 | 
						|
 | 
						|
private:
 | 
						|
	struct InvitePeek {
 | 
						|
		QString hash;
 | 
						|
		TimeId expires = 0;
 | 
						|
	};
 | 
						|
 | 
						|
	auto unavailableReasons() const
 | 
						|
		-> const std::vector<Data::UnavailableReason> & override;
 | 
						|
	bool canEditLastAdmin(not_null<UserData*> user) const;
 | 
						|
 | 
						|
	Flags _flags = ChannelDataFlags(Flag::Forbidden);
 | 
						|
 | 
						|
	PtsWaiter _ptsWaiter;
 | 
						|
 | 
						|
	int _membersCount = -1;
 | 
						|
	int _adminsCount = 1;
 | 
						|
	int _restrictedCount = 0;
 | 
						|
	int _kickedCount = 0;
 | 
						|
	int _pendingRequestsCount = 0;
 | 
						|
	std::vector<UserId> _recentRequesters;
 | 
						|
	MsgId _availableMinId = 0;
 | 
						|
 | 
						|
	RestrictionFlags _defaultRestrictions;
 | 
						|
	AdminRightFlags _adminRights;
 | 
						|
	RestrictionFlags _restrictions;
 | 
						|
	TimeId _restrictedUntil;
 | 
						|
 | 
						|
	std::vector<Data::UnavailableReason> _unavailableReasons;
 | 
						|
	std::unique_ptr<InvitePeek> _invitePeek;
 | 
						|
	QString _inviteLink;
 | 
						|
	std::optional<ChannelData*> _linkedChat;
 | 
						|
 | 
						|
	std::vector<QString> _allowedReactions;
 | 
						|
 | 
						|
	std::unique_ptr<Data::GroupCall> _call;
 | 
						|
	PeerId _callDefaultJoinAs = 0;
 | 
						|
 | 
						|
	int _slowmodeSeconds = 0;
 | 
						|
	TimeId _slowmodeLastMessage = 0;
 | 
						|
 | 
						|
	rpl::lifetime _lifetime;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
namespace Data {
 | 
						|
 | 
						|
void ApplyMigration(
 | 
						|
	not_null<ChatData*> chat,
 | 
						|
	not_null<ChannelData*> channel);
 | 
						|
 | 
						|
void ApplyChannelUpdate(
 | 
						|
	not_null<ChannelData*> channel,
 | 
						|
	const MTPDupdateChatDefaultBannedRights &update);
 | 
						|
 | 
						|
void ApplyChannelUpdate(
 | 
						|
	not_null<ChannelData*> channel,
 | 
						|
	const MTPDchannelFull &update);
 | 
						|
 | 
						|
} // namespace Data
 |