/* 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 "history/history_item.h" namespace Api { struct SendAction; } // namespace Api namespace HistoryView { class Message; } // namespace HistoryView struct HistoryMessageEdited; struct HistoryMessageReply; struct HistoryMessageViews; [[nodiscard]] Fn HistoryDependentItemCallback( not_null item); [[nodiscard]] MTPDmessage::Flags NewMessageFlags(not_null peer); [[nodiscard]] MTPDmessage_ClientFlags NewMessageClientFlags(); [[nodiscard]] MsgId LookupReplyToTop( not_null history, MsgId replyToId); [[nodiscard]] MTPMessageReplyHeader NewMessageReplyHeader( const Api::SendAction &action); [[nodiscard]] QString GetErrorTextForSending( not_null peer, const HistoryItemsList &items, bool ignoreSlowmodeCountdown = false); [[nodiscard]] QString GetErrorTextForSending( not_null peer, const HistoryItemsList &items, const TextWithTags &comment, bool ignoreSlowmodeCountdown = false); void FastShareMessage(not_null item); class HistoryMessage final : public HistoryItem { public: HistoryMessage( not_null history, const MTPDmessage &data, MTPDmessage_ClientFlags clientFlags); HistoryMessage( not_null history, const MTPDmessageService &data, MTPDmessage_ClientFlags clientFlags); HistoryMessage( not_null history, MsgId id, MTPDmessage::Flags flags, MTPDmessage_ClientFlags clientFlags, TimeId date, PeerId from, const QString &postAuthor, not_null original); // local forwarded HistoryMessage( not_null history, MsgId id, MTPDmessage::Flags flags, MTPDmessage_ClientFlags clientFlags, MsgId replyTo, UserId viaBotId, TimeId date, PeerId from, const QString &postAuthor, const TextWithEntities &textWithEntities); // local message HistoryMessage( not_null history, MsgId id, MTPDmessage::Flags flags, MTPDmessage_ClientFlags clientFlags, MsgId replyTo, UserId viaBotId, TimeId date, PeerId from, const QString &postAuthor, not_null document, const TextWithEntities &caption, const MTPReplyMarkup &markup); // local document HistoryMessage( not_null history, MsgId id, MTPDmessage::Flags flags, MTPDmessage_ClientFlags clientFlags, MsgId replyTo, UserId viaBotId, TimeId date, PeerId from, const QString &postAuthor, not_null photo, const TextWithEntities &caption, const MTPReplyMarkup &markup); // local photo HistoryMessage( not_null history, MsgId id, MTPDmessage::Flags flags, MTPDmessage_ClientFlags clientFlags, MsgId replyTo, UserId viaBotId, TimeId date, PeerId from, const QString &postAuthor, not_null game, const MTPReplyMarkup &markup); // local game void refreshMedia(const MTPMessageMedia *media); void refreshSentMedia(const MTPMessageMedia *media); void returnSavedMedia() override; void setMedia(const MTPMessageMedia &media); [[nodiscard]] static std::unique_ptr CreateMedia( not_null item, const MTPMessageMedia &media); [[nodiscard]] bool allowsForward() const override; [[nodiscard]] bool allowsSendNow() const override; [[nodiscard]] bool allowsEdit(TimeId now) const override; [[nodiscard]] bool uploading() const; [[nodiscard]] bool hideEditedBadge() const { return (_flags & MTPDmessage::Flag::f_edit_hide); } void setViewsCount(int count) override; void setForwardsCount(int count) override; void setReplies(const MTPMessageReplies &data) override; void clearReplies() override; void changeRepliesCount(int delta, PeerId replier) override; void setReplyToTop(MsgId replyToTop) override; void setPostAuthor(const QString &author) override; void setRealId(MsgId newId) override; void incrementReplyToTopCounter() override; void dependencyItemRemoved(HistoryItem *dependency) override; [[nodiscard]] QString notificationHeader() const override; void applyEdition(const MTPDmessage &message) override; void applyEdition(const MTPDmessageService &message) override; void updateSentContent( const TextWithEntities &textWithEntities, const MTPMessageMedia *media) override; void updateReplyMarkup(const MTPReplyMarkup *markup) override { setReplyMarkup(markup); } void updateForwardedInfo(const MTPMessageFwdHeader *fwd) override; void contributeToSlowmode(TimeId realDate = 0) override; void addToUnreadMentions(UnreadMentionType type) override; void destroyHistoryEntry() override; [[nodiscard]] Storage::SharedMediaTypesMask sharedMediaTypes() const override; void setText(const TextWithEntities &textWithEntities) override; [[nodiscard]] Ui::Text::IsolatedEmoji isolatedEmoji() const override; [[nodiscard]] TextWithEntities originalText() const override; [[nodiscard]] TextForMimeData clipboardText() const override; [[nodiscard]] bool textHasLinks() const override; [[nodiscard]] int viewsCount() const override; [[nodiscard]] int repliesCount() const override; [[nodiscard]] bool repliesAreComments() const override; [[nodiscard]] bool externalReply() const override; [[nodiscard]] MsgId repliesInboxReadTill() const override; void setRepliesInboxReadTill(MsgId readTillId) override; [[nodiscard]] MsgId computeRepliesInboxReadTillFull() const override; [[nodiscard]] MsgId repliesOutboxReadTill() const override; void setRepliesOutboxReadTill(MsgId readTillId) override; [[nodiscard]] MsgId computeRepliesOutboxReadTillFull() const override; void setRepliesMaxId(MsgId maxId) override; void setRepliesPossibleMaxId(MsgId possibleMaxId) override; [[nodiscard]] bool areRepliesUnread() const override; [[nodiscard]] FullMsgId commentsItemId() const override; void setCommentsItemId(FullMsgId id) override; bool updateDependencyItem() override; [[nodiscard]] MsgId dependencyMsgId() const override { return replyToId(); } // dynamic_cast optimization. [[nodiscard]] HistoryMessage *toHistoryMessage() override { return this; } [[nodiscard]] const HistoryMessage *toHistoryMessage() const override { return this; } [[nodiscard]] std::unique_ptr createView( not_null delegate, HistoryView::Element *replacing = nullptr) override; ~HistoryMessage(); private: void setEmptyText(); [[nodiscard]] bool isTooOldForEdit(TimeId now) const; [[nodiscard]] bool isLegacyMessage() const { return _flags & MTPDmessage::Flag::f_legacy; } [[nodiscard]] bool checkCommentsLinkedChat(ChannelId id) const; void clearIsolatedEmoji(); void checkIsolatedEmoji(); // For an invoice button we replace the button text with a "Receipt" key. // It should show the receipt for the payed invoice. Still let mobile apps do that. void replaceBuyWithReceiptInMarkup(); void setReplyMarkup(const MTPReplyMarkup *markup); struct CreateConfig; void createComponentsHelper(MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, const QString &postAuthor, const MTPReplyMarkup &markup); void createComponents(const CreateConfig &config); void setupForwardedComponent(const CreateConfig &config); void changeReplyToTopCounter( not_null reply, int delta); void refreshRepliesText( not_null views, bool forceResize = false); static void FillForwardedInfo( CreateConfig &config, const MTPDmessageFwdHeader &data); [[nodiscard]] bool generateLocalEntitiesByReply() const; [[nodiscard]] TextWithEntities withLocalEntities( const TextWithEntities &textWithEntities) const; void reapplyText(); QString _timeText; int _timeWidth = 0; mutable int32 _fromNameVersion = 0; friend class HistoryView::Element; friend class HistoryView::Message; };