tdesktop/Telegram/SourceFiles/history/view/history_view_element.h

292 lines
7.7 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 "history/view/history_view_object.h"
#include "base/runtime_composer.h"
#include "base/flags.h"
class HistoryBlock;
class HistoryItem;
class HistoryMessage;
class HistoryService;
class HistoryMedia;
class HistoryWebPage;
namespace HistoryView {
enum class PointState : char;
enum class InfoDisplayType : char;
struct StateRequest;
struct TextState;
enum class Context : char {
History,
Feed,
AdminLog,
ContactPreview
};
class Element;
class ElementDelegate {
public:
virtual Context elementContext() = 0;
virtual std::unique_ptr<Element> elementCreate(
not_null<HistoryMessage*> message) = 0;
virtual std::unique_ptr<Element> elementCreate(
not_null<HistoryService*> message) = 0;
virtual bool elementUnderCursor(not_null<const Element*> view) = 0;
virtual void elementAnimationAutoplayAsync(
not_null<const Element*> element) = 0;
virtual TimeMs elementHighlightTime(
not_null<const Element*> element) = 0;
virtual bool elementInSelectionMode() = 0;
};
TextSelection UnshiftItemSelection(
TextSelection selection,
uint16 byLength);
TextSelection ShiftItemSelection(
TextSelection selection,
uint16 byLength);
TextSelection UnshiftItemSelection(
TextSelection selection,
const Text &byText);
TextSelection ShiftItemSelection(
TextSelection selection,
const Text &byText);
// Any HistoryView::Element can have this Component for
// displaying the unread messages bar above the message.
struct UnreadBar : public RuntimeComponent<UnreadBar, Element> {
void init(int newCount);
static int height();
static int marginTop();
void paint(Painter &p, int y, int w) const;
static constexpr auto kCountUnknown = std::numeric_limits<int>::max();
QString text;
int width = 0;
int count = 0;
// If unread bar is freezed the new messages do not
// increment the counter displayed by this bar.
//
// It happens when we've opened the conversation and
// we've seen the bar and new messages are marked as read
// as soon as they are added to the chat history.
bool freezed = false;
};
// Any HistoryView::Element can have this Component for
// displaying the day mark above the message.
struct DateBadge : public RuntimeComponent<DateBadge, Element> {
void init(const QDateTime &date);
int height() const;
void paint(Painter &p, int y, int w) const;
QString text;
int width = 0;
};
class Element
: public Object
, public RuntimeComposer<Element>
, public ClickHandlerHost {
public:
Element(
not_null<ElementDelegate*> delegate,
not_null<HistoryItem*> data);
enum class Flag : uchar {
NeedsResize = 0x01,
AttachedToPrevious = 0x02,
AttachedToNext = 0x04,
HiddenByGroup = 0x08,
};
using Flags = base::flags<Flag>;
friend inline constexpr auto is_flag_type(Flag) { return true; }
not_null<ElementDelegate*> delegate() const;
not_null<HistoryItem*> data() const;
HistoryMedia *media() const;
Context context() const;
void refreshDataId();
QDateTime dateTime() const;
int y() const;
void setY(int y);
virtual int marginTop() const = 0;
virtual int marginBottom() const = 0;
void setPendingResize();
bool pendingResize() const;
bool isUnderCursor() const;
bool isAttachedToPrevious() const;
bool isAttachedToNext() const;
int skipBlockWidth() const;
int skipBlockHeight() const;
QString skipBlock() const;
virtual int infoWidth() const;
bool isHiddenByGroup() const;
virtual bool isHidden() const;
// For blocks context this should be called only from recountAttachToPreviousInBlocks().
void setAttachToPrevious(bool attachToNext);
// For blocks context this should be called only from recountAttachToPreviousInBlocks()
// of the next item or when the next item is removed through nextInBlocksRemoved() call.
void setAttachToNext(bool attachToNext);
// For blocks context this should be called only from recountDisplayDate().
void setDisplayDate(bool displayDate);
bool computeIsAttachToPrevious(not_null<Element*> previous);
void setUnreadBarCount(int count);
void destroyUnreadBar();
// marks the unread bar as freezed so that unread
// messages count will not change for this bar
// when the new messages arrive in this chat history
void setUnreadBarFreezed();
int displayedDateHeight() const;
bool displayDate() const;
bool isInOneDayWithPrevious() const;
virtual void draw(
Painter &p,
QRect clip,
TextSelection selection,
TimeMs ms) const = 0;
[[nodiscard]] virtual PointState pointState(QPoint point) const = 0;
[[nodiscard]] virtual TextState textState(
QPoint point,
StateRequest request) const = 0;
virtual void updatePressed(QPoint point) = 0;
virtual void drawInfo(
Painter &p,
int right,
int bottom,
int width,
bool selected,
InfoDisplayType type) const;
virtual bool pointInTime(
int right,
int bottom,
QPoint point,
InfoDisplayType type) const;
virtual TextWithEntities selectedText(
TextSelection selection) const = 0;
[[nodiscard]] virtual TextSelection adjustSelection(
TextSelection selection,
TextSelectType type) const;
// ClickHandlerHost interface.
void clickHandlerActiveChanged(
const ClickHandlerPtr &handler,
bool active) override;
void clickHandlerPressedChanged(
const ClickHandlerPtr &handler,
bool pressed) override;
// hasFromPhoto() returns true even if we don't display the photo
// but we need to skip a place at the left side for this photo
virtual bool hasFromPhoto() const;
virtual bool displayFromPhoto() const;
virtual bool hasFromName() const;
virtual bool displayFromName() const;
virtual bool displayForwardedFrom() const;
virtual bool hasOutLayout() const;
virtual bool drawBubble() const;
virtual bool hasBubble() const;
virtual bool hasFastReply() const;
virtual bool displayFastReply() const;
virtual bool displayRightAction() const;
virtual void drawRightAction(
Painter &p,
int left,
int top,
int outerWidth) const;
virtual ClickHandlerPtr rightActionLink() const;
virtual bool displayEditedBadge() const;
virtual TimeId displayedEditDate() const;
virtual bool hasVisibleText() const;
// Legacy blocks structure.
HistoryBlock *block();
const HistoryBlock *block() const;
void attachToBlock(not_null<HistoryBlock*> block, int index);
void removeFromBlock();
void refreshInBlock();
void setIndexInBlock(int index);
int indexInBlock() const;
Element *previousInBlocks() const;
Element *nextInBlocks() const;
void previousInBlocksChanged();
void nextInBlocksRemoved();
virtual ~Element();
protected:
void paintHighlight(
Painter &p,
int geometryHeight) const;
virtual void refreshDataIdHook();
private:
// This should be called only from previousInBlocksChanged()
// to add required bits to the Composer mask
// after that always use Has<DateBadge>().
void recountDisplayDateInBlocks();
// This should be called only from previousInBlocksChanged() or when
// DateBadge or UnreadBar bit is changed in the Composer mask
// then the result should be cached in a client side flag
// MTPDmessage_ClientFlag::f_attach_to_previous.
void recountAttachToPreviousInBlocks();
QSize countOptimalSize() final override;
QSize countCurrentSize(int newWidth) final override;
virtual QSize performCountOptimalSize() = 0;
virtual QSize performCountCurrentSize(int newWidth) = 0;
void refreshMedia();
const not_null<ElementDelegate*> _delegate;
const not_null<HistoryItem*> _data;
std::unique_ptr<HistoryMedia> _media;
const QDateTime _dateTime;
int _y = 0;
Context _context = Context();
Flags _flags = Flag::NeedsResize;
HistoryBlock *_block = nullptr;
int _indexInBlock = -1;
};
} // namespace HistoryView