tdesktop/Telegram/SourceFiles/history/history_item.h

471 lines
13 KiB
C
Raw Normal View History

/*
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
2017-04-06 14:38:10 +00:00
#include "base/runtime_composer.h"
2017-08-31 16:28:58 +00:00
#include "base/flags.h"
2017-12-13 18:10:48 +00:00
#include "base/value_ordering.h"
#include "data/data_media_types.h"
enum class UnreadMentionType;
struct HistoryMessageReplyMarkup;
class ReplyKeyboard;
class HistoryMessage;
namespace base {
template <typename Enum>
class enum_mask;
} // namespace base
namespace Storage {
enum class SharedMediaType : signed char;
using SharedMediaTypesMask = base::enum_mask<SharedMediaType>;
} // namespace Storage
namespace Ui {
class RippleAnimation;
} // namespace Ui
namespace style {
struct BotKeyboardButton;
struct RippleAnimation;
} // namespace style
2018-01-09 17:08:31 +00:00
namespace Data {
struct MessagePosition;
class Media;
2018-01-09 17:08:31 +00:00
} // namespace Data
namespace Window {
class SessionController;
} // namespace Window
namespace HistoryView {
struct TextState;
struct StateRequest;
enum class CursorState : char;
enum class PointState : char;
enum class Context : char;
class ElementDelegate;
} // namespace HistoryView
struct HiddenSenderInfo;
class History;
class HistoryItem : public RuntimeComposer<HistoryItem> {
public:
static not_null<HistoryItem*> Create(
not_null<History*> history,
const MTPMessage &message,
MTPDmessage_ClientFlags clientFlags);
2018-02-05 20:19:51 +00:00
struct Destroyer {
void operator()(HistoryItem *value);
};
virtual void dependencyItemRemoved(HistoryItem *dependency) {
}
virtual bool updateDependencyItem() {
return true;
}
virtual MsgId dependencyMsgId() const {
return 0;
}
[[nodiscard]] virtual bool notificationReady() const {
return true;
}
[[nodiscard]] PeerData *specialNotificationPeer() const;
[[nodiscard]] UserData *viaBot() const;
[[nodiscard]] UserData *getMessageBot() const;
[[nodiscard]] bool isHistoryEntry() const;
[[nodiscard]] bool isAdminLogEntry() const;
[[nodiscard]] bool isFromScheduled() const;
[[nodiscard]] bool isScheduled() const;
void addLogEntryOriginal(
WebPageId localId,
const QString &label,
const TextWithEntities &content);
not_null<History*> history() const {
return _history;
}
not_null<PeerData*> from() const {
return _from;
}
HistoryView::Element *mainView() const {
return _mainView;
}
void setMainView(not_null<HistoryView::Element*> view) {
_mainView = view;
}
2018-01-18 13:59:22 +00:00
void refreshMainView();
void clearMainView();
void removeMainView();
void destroy();
[[nodiscard]] bool out() const {
return _flags & MTPDmessage::Flag::f_out;
}
[[nodiscard]] bool isPinned() const {
return _flags & MTPDmessage::Flag::f_pinned;
}
[[nodiscard]] bool unread() const;
[[nodiscard]] bool showNotification() const;
void markClientSideAsRead();
[[nodiscard]] bool mentionsMe() const;
[[nodiscard]] bool isUnreadMention() const;
[[nodiscard]] bool isUnreadMedia() const;
[[nodiscard]] bool hasUnreadMediaFlag() const;
void markMediaRead();
void setIsPinned(bool isPinned);
// For edit media in history_message.
virtual void returnSavedMedia() {};
void savePreviousMedia() {
_savedLocalEditMediaData = {
originalText(),
_media->clone(this),
};
}
[[nodiscard]] bool isEditingMedia() const {
return _savedLocalEditMediaData.media != nullptr;
}
void clearSavedMedia() {
_savedLocalEditMediaData = {};
}
// Zero result means this message is not self-destructing right now.
virtual crl::time getSelfDestructIn(crl::time now) {
return 0;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool definesReplyKeyboard() const;
[[nodiscard]] MTPDreplyKeyboardMarkup::Flags replyKeyboardFlags() const;
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool hasSwitchInlineButton() const {
return _clientFlags & MTPDmessage_ClientFlag::f_has_switch_inline_button;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool hasTextLinks() const {
return _clientFlags & MTPDmessage_ClientFlag::f_has_text_links;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool isGroupEssential() const {
return _clientFlags & MTPDmessage_ClientFlag::f_is_group_essential;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool isLocalUpdateMedia() const {
return _clientFlags & MTPDmessage_ClientFlag::f_is_local_update_media;
}
void setIsLocalUpdateMedia(bool flag) {
if (flag) {
_clientFlags |= MTPDmessage_ClientFlag::f_is_local_update_media;
} else {
_clientFlags &= ~MTPDmessage_ClientFlag::f_is_local_update_media;
}
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool isGroupMigrate() const {
2019-01-22 07:50:21 +00:00
return isGroupEssential() && isEmpty();
}
2019-08-02 18:19:14 +00:00
[[nodiscard]] bool isIsolatedEmoji() const {
return _clientFlags & MTPDmessage_ClientFlag::f_isolated_emoji;
2019-08-01 21:14:19 +00:00
}
[[nodiscard]] bool hasViews() const {
return _flags & MTPDmessage::Flag::f_views;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool isPost() const {
return _flags & MTPDmessage::Flag::f_post;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool isSilent() const {
return _flags & MTPDmessage::Flag::f_silent;
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool isSending() const {
return _clientFlags & MTPDmessage_ClientFlag::f_sending;
2019-07-17 14:34:39 +00:00
}
2019-08-01 21:14:19 +00:00
[[nodiscard]] bool hasFailed() const {
return _clientFlags & MTPDmessage_ClientFlag::f_failed;
2019-07-17 14:34:39 +00:00
}
void sendFailed();
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual int viewsCount() const {
return hasViews() ? 1 : -1;
}
[[nodiscard]] virtual int repliesCount() const {
return 0;
}
2020-09-03 07:19:02 +00:00
[[nodiscard]] virtual bool repliesAreComments() const {
return false;
}
[[nodiscard]] virtual bool externalReply() const {
return false;
}
[[nodiscard]] virtual MsgId repliesInboxReadTill() const {
return MsgId(0);
2020-09-03 08:21:31 +00:00
}
virtual void setRepliesInboxReadTill(MsgId readTillId) {
2020-09-03 08:21:31 +00:00
}
[[nodiscard]] virtual MsgId computeRepliesInboxReadTillFull() const {
2020-09-15 08:39:39 +00:00
return MsgId(0);
}
[[nodiscard]] virtual MsgId repliesOutboxReadTill() const {
return MsgId(0);
2020-09-15 08:39:39 +00:00
}
virtual void setRepliesOutboxReadTill(MsgId readTillId) {
2020-09-15 08:39:39 +00:00
}
[[nodiscard]] virtual MsgId computeRepliesOutboxReadTillFull() const {
return MsgId(0);
}
virtual void setRepliesMaxId(MsgId maxId) {
}
virtual void setRepliesPossibleMaxId(MsgId possibleMaxId) {
}
[[nodiscard]] virtual bool areRepliesUnread() const {
2020-09-15 08:39:39 +00:00
return false;
}
[[nodiscard]] virtual FullMsgId commentsItemId() const {
return FullMsgId();
}
virtual void setCommentsItemId(FullMsgId id) {
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual bool needCheck() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual bool serviceMsg() const {
return false;
}
virtual void applyEdition(const MTPDmessage &message) {
}
virtual void applyEdition(const MTPDmessageService &message) {
}
2019-04-08 09:16:45 +00:00
void applyEditionToHistoryCleared();
virtual void updateSentContent(
const TextWithEntities &textWithEntities,
const MTPMessageMedia *media) {
}
virtual void updateReplyMarkup(const MTPReplyMarkup *markup) {
}
virtual void updateForwardedInfo(const MTPMessageFwdHeader *fwd) {
}
2019-08-20 09:42:13 +00:00
virtual void contributeToSlowmode(TimeId realDate = 0) {
}
virtual void addToUnreadMentions(UnreadMentionType type);
virtual void destroyHistoryEntry() {
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual Storage::SharedMediaTypesMask sharedMediaTypes() const = 0;
2021-02-02 16:38:30 +00:00
virtual void applySentMessage(const MTPDmessage &data);
virtual void applySentMessage(
const QString &text,
const MTPDupdateShortSentMessage &data,
bool wasAlready);
void indexAsNewItem();
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual QString notificationHeader() const {
return QString();
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual QString notificationText() const;
enum class DrawInDialog {
Normal,
WithoutSender,
};
// Returns text with link-start and link-end commands for service-color highlighting.
// Example: "[link1-start]You:[link1-end] [link1-start]Photo,[link1-end] caption text"
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual QString inDialogsText(DrawInDialog way) const;
[[nodiscard]] virtual QString inReplyText() const {
return inDialogsText(DrawInDialog::WithoutSender);
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual Ui::Text::IsolatedEmoji isolatedEmoji() const;
[[nodiscard]] virtual TextWithEntities originalText() const {
return TextWithEntities();
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual TextForMimeData clipboardText() const {
return TextForMimeData();
}
virtual void setViewsCount(int count) {
}
virtual void setForwardsCount(int count) {
}
2020-09-03 07:19:02 +00:00
virtual void setReplies(const MTPMessageReplies &data) {
}
virtual void clearReplies() {
}
virtual void changeRepliesCount(int delta, PeerId replier) {
}
virtual void setReplyToTop(MsgId replyToTop) {
}
virtual void setPostAuthor(const QString &author) {
}
virtual void setRealId(MsgId newId);
virtual void incrementReplyToTopCounter() {
}
void drawInDialog(
Painter &p,
const QRect &r,
bool active,
bool selected,
DrawInDialog way,
const HistoryItem *&cacheFor,
2019-06-12 13:26:04 +00:00
Ui::Text::String &cache) const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] bool emptyText() const {
return _text.isEmpty();
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] bool canPin() const;
[[nodiscard]] bool canStopPoll() const;
[[nodiscard]] virtual bool allowsSendNow() const;
[[nodiscard]] virtual bool allowsForward() const;
[[nodiscard]] virtual bool allowsEdit(TimeId now) const;
[[nodiscard]] bool canDelete() const;
[[nodiscard]] bool canDeleteForEveryone(TimeId now) const;
[[nodiscard]] bool suggestReport() const;
[[nodiscard]] bool suggestBanReport() const;
[[nodiscard]] bool suggestDeleteAllReport() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] bool hasDirectLink() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] ChannelId channelId() const;
[[nodiscard]] FullMsgId fullId() const {
return FullMsgId(channelId(), id);
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] Data::MessagePosition position() const;
[[nodiscard]] TimeId date() const;
[[nodiscard]] static TimeId NewMessageDate(TimeId scheduled);
2019-08-12 16:33:36 +00:00
[[nodiscard]] Data::Media *media() const {
return _media.get();
}
virtual void setText(const TextWithEntities &textWithEntities) {
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual bool textHasLinks() const {
return false;
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual HistoryMessage *toHistoryMessage() { // dynamic_cast optimize
return nullptr;
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual const HistoryMessage *toHistoryMessage() const { // dynamic_cast optimize
return nullptr;
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] MsgId replyToId() const;
[[nodiscard]] MsgId replyToTop() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] not_null<PeerData*> author() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] TimeId dateOriginal() const;
[[nodiscard]] PeerData *senderOriginal() const;
[[nodiscard]] const HiddenSenderInfo *hiddenForwardedInfo() const;
[[nodiscard]] not_null<PeerData*> fromOriginal() const;
[[nodiscard]] QString authorOriginal() const;
[[nodiscard]] MsgId idOriginal() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] bool isEmpty() const;
2017-12-13 18:10:48 +00:00
2019-08-12 16:33:36 +00:00
[[nodiscard]] MessageGroupId groupId() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] const HistoryMessageReplyMarkup *inlineReplyMarkup() const {
2019-05-25 14:42:01 +00:00
return const_cast<HistoryItem*>(this)->inlineReplyMarkup();
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] const ReplyKeyboard *inlineReplyKeyboard() const {
2019-05-25 14:42:01 +00:00
return const_cast<HistoryItem*>(this)->inlineReplyKeyboard();
}
2019-08-12 16:33:36 +00:00
[[nodiscard]] HistoryMessageReplyMarkup *inlineReplyMarkup();
[[nodiscard]] ReplyKeyboard *inlineReplyKeyboard();
2019-05-25 14:42:01 +00:00
[[nodiscard]] ChannelData *discussionPostOriginalSender() const;
[[nodiscard]] bool isDiscussionPost() const;
2020-09-15 08:39:39 +00:00
[[nodiscard]] HistoryItem *lookupDiscussionPostOriginal() const;
[[nodiscard]] PeerData *displayFrom() const;
2019-08-12 16:33:36 +00:00
[[nodiscard]] virtual std::unique_ptr<HistoryView::Element> createView(
not_null<HistoryView::ElementDelegate*> delegate,
HistoryView::Element *replacing = nullptr) = 0;
void updateDate(TimeId newDate);
[[nodiscard]] bool canUpdateDate() const;
[[nodiscard]] TimeId ttlDestroyAt() const {
return _ttlDestroyAt;
}
virtual ~HistoryItem();
2019-08-12 16:33:36 +00:00
MsgId id;
protected:
HistoryItem(
not_null<History*> history,
MsgId id,
MTPDmessage::Flags flags,
MTPDmessage_ClientFlags clientFlags,
TimeId date,
PeerId from);
virtual void markMediaAsReadHook() {
}
void applyServiceDateEdition(const MTPDmessageService &data);
void finishEdition(int oldKeyboardTop);
void finishEditionToEmpty();
const not_null<History*> _history;
not_null<PeerData*> _from;
MTPDmessage::Flags _flags = 0;
MTPDmessage_ClientFlags _clientFlags = 0;
2019-01-15 11:57:45 +00:00
void invalidateChatListEntry();
void setGroupId(MessageGroupId groupId);
void applyTTL(const MTPDmessage &data);
void applyTTL(const MTPDmessageService &data);
void applyTTL(TimeId destroyAt);
2019-06-12 13:26:04 +00:00
Ui::Text::String _text = { st::msgMinWidth };
int _textWidth = -1;
int _textHeight = 0;
struct SavedMediaData {
TextWithEntities text;
std::unique_ptr<Data::Media> media;
};
SavedMediaData _savedLocalEditMediaData;
std::unique_ptr<Data::Media> _media;
private:
TimeId _date = 0;
TimeId _ttlDestroyAt = 0;
HistoryView::Element *_mainView = nullptr;
friend class HistoryView::Element;
2019-05-31 21:51:57 +00:00
MessageGroupId _groupId = MessageGroupId();
};
QDateTime ItemDateTime(not_null<const HistoryItem*> item);
QString ItemDateText(not_null<const HistoryItem*> item, bool isUntilOnline);
bool IsItemScheduledUntilOnline(not_null<const HistoryItem*> item);
ClickHandlerPtr goToMessageClickHandler(
not_null<PeerData*> peer,
MsgId msgId,
FullMsgId returnToId = FullMsgId());
ClickHandlerPtr goToMessageClickHandler(
not_null<HistoryItem*> item,
FullMsgId returnToId = FullMsgId());