Add to each history item its own width value.

This commit is contained in:
John Preston 2017-06-21 11:53:14 +03:00
parent 839e59075d
commit 507b7d7193
7 changed files with 206 additions and 219 deletions

View File

@ -116,7 +116,7 @@ msgReplyBarPos: point(1px, 0px);
msgReplyBarSize: size(2px, 36px); msgReplyBarSize: size(2px, 36px);
msgReplyBarSkip: 10px; msgReplyBarSkip: 10px;
msgServicePadding: margins(12px, 3px, 12px, 4px); msgServicePadding: margins(12px, 3px, 12px, 4px);
msgServiceMargin: margins(10px, 10px, 80px, 2px); msgServiceMargin: margins(10px, 10px, 10px, 2px);
msgDateSpace: 12px; msgDateSpace: 12px;
msgDateDelta: point(2px, 5px); msgDateDelta: point(2px, 5px);

View File

@ -112,8 +112,10 @@ void FixedBar::setAnimatingMode(bool enabled) {
} }
void FixedBar::paintEvent(QPaintEvent *e) { void FixedBar::paintEvent(QPaintEvent *e) {
Painter p(this); if (!_animatingMode) {
p.fillRect(e->rect(), st::topBarBg); Painter p(this);
p.fillRect(e->rect(), st::topBarBg);
}
} }
void FixedBar::mousePressEvent(QMouseEvent *e) { void FixedBar::mousePressEvent(QMouseEvent *e) {

View File

@ -594,8 +594,8 @@ TextSelection shiftSelection(TextSelection selection, const Text &byText) {
HistoryItem::HistoryItem(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime msgDate, int32 from) : HistoryElement() HistoryItem::HistoryItem(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime msgDate, int32 from) : HistoryElement()
, id(msgId) , id(msgId)
, date(msgDate) , date(msgDate)
, _from(from ? App::user(from) : history->peer)
, _history(history) , _history(history)
, _from(from ? App::user(from) : history->peer)
, _flags(flags | MTPDmessage_ClientFlag::f_pending_init_dimensions | MTPDmessage_ClientFlag::f_pending_resize) , _flags(flags | MTPDmessage_ClientFlag::f_pending_init_dimensions | MTPDmessage_ClientFlag::f_pending_resize)
, _authorNameVersion(author()->nameVersion) { , _authorNameVersion(author()->nameVersion) {
} }

View File

@ -476,7 +476,7 @@ TextSelection shiftSelection(TextSelection selection, const Text &byText);
class HistoryItem : public HistoryElement, public RuntimeComposer, public ClickHandlerHost { class HistoryItem : public HistoryElement, public RuntimeComposer, public ClickHandlerHost {
public: public:
int resizeGetHeight(int width) { int resizeGetHeight(int newWidth) {
if (_flags & MTPDmessage_ClientFlag::f_pending_init_dimensions) { if (_flags & MTPDmessage_ClientFlag::f_pending_init_dimensions) {
_flags &= ~MTPDmessage_ClientFlag::f_pending_init_dimensions; _flags &= ~MTPDmessage_ClientFlag::f_pending_init_dimensions;
initDimensions(); initDimensions();
@ -484,9 +484,10 @@ public:
if (_flags & MTPDmessage_ClientFlag::f_pending_resize) { if (_flags & MTPDmessage_ClientFlag::f_pending_resize) {
_flags &= ~MTPDmessage_ClientFlag::f_pending_resize; _flags &= ~MTPDmessage_ClientFlag::f_pending_resize;
} }
return resizeGetHeight_(width); _width = newWidth;
return resizeContentGetHeight();
} }
virtual void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const = 0; virtual void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const = 0;
virtual void dependencyItemRemoved(HistoryItem *dependency) { virtual void dependencyItemRemoved(HistoryItem *dependency) {
} }
@ -501,7 +502,7 @@ public:
} }
UserData *viaBot() const { UserData *viaBot() const {
if (const HistoryMessageVia *via = Get<HistoryMessageVia>()) { if (auto via = Get<HistoryMessageVia>()) {
return via->_bot; return via->_bot;
} }
return nullptr; return nullptr;
@ -871,6 +872,10 @@ public:
return _text.isEmpty() && !_media && !Has<HistoryMessageLogEntryOriginal>(); return _text.isEmpty() && !_media && !Has<HistoryMessageLogEntryOriginal>();
} }
int width() const {
return _width;
}
void clipCallback(Media::Clip::Notification notification); void clipCallback(Media::Clip::Notification notification);
void audioTrackUpdated(); void audioTrackUpdated();
@ -886,18 +891,18 @@ protected:
// called from resizeGetHeight() when MTPDmessage_ClientFlag::f_pending_init_dimensions is set // called from resizeGetHeight() when MTPDmessage_ClientFlag::f_pending_init_dimensions is set
virtual void initDimensions() = 0; virtual void initDimensions() = 0;
virtual int resizeGetHeight_(int width) = 0; virtual int resizeContentGetHeight() = 0;
void finishEdition(int oldKeyboardTop); void finishEdition(int oldKeyboardTop);
void finishEditionToEmpty(); void finishEditionToEmpty();
PeerData *_from; gsl::not_null<History*> _history;
History *_history; gsl::not_null<PeerData*> _from;
HistoryBlock *_block = nullptr; HistoryBlock *_block = nullptr;
int _indexInBlock = -1; int _indexInBlock = -1;
MTPDmessage::Flags _flags; MTPDmessage::Flags _flags = 0;
mutable int32 _authorNameVersion; mutable int32 _authorNameVersion = 0;
HistoryItem *previousItem() const { HistoryItem *previousItem() const {
if (_block && _indexInBlock >= 0) { if (_block && _indexInBlock >= 0) {
@ -974,6 +979,7 @@ protected:
private: private:
int _y = 0; int _y = 0;
int _width = 0;
}; };
@ -985,8 +991,8 @@ template <typename T>
class HistoryItemInstantiated { class HistoryItemInstantiated {
public: public:
template <typename ...Args> template <typename ...Args>
static T *_create(Args &&... args) { static gsl::not_null<T*> _create(Args &&... args) {
T *result = new T(std::forward<Args>(args)...); auto result = new T(std::forward<Args>(args)...);
result->finishCreate(); result->finishCreate();
return result; return result;
} }

View File

@ -35,6 +35,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
namespace { namespace {
constexpr auto kPinnedMessageTextLimit = 16;
TextParseOptions _historySrvOptions = { TextParseOptions _historySrvOptions = {
TextParseLinks | TextParseMentions | TextParseHashtags/* | TextParseMultiline*/ | TextParseRichText, // flags TextParseLinks | TextParseMentions | TextParseHashtags/* | TextParseMultiline*/ | TextParseRichText, // flags
0, // maxw 0, // maxw
@ -55,10 +57,8 @@ ApiWrap::RequestMessageDataCallback historyDependentItemCallback(const FullMsgId
}; };
} }
constexpr auto kPinnedMessageTextLimit = 16;
style::color fromNameFg(int index) { style::color fromNameFg(int index) {
t_assert(index >= 0 && index < 8); Expects(index >= 0 && index < 8);
style::color colors[] = { style::color colors[] = {
st::historyPeer1NameFg, st::historyPeer1NameFg,
st::historyPeer2NameFg, st::historyPeer2NameFg,
@ -73,7 +73,7 @@ style::color fromNameFg(int index) {
} }
style::color fromNameFgSelected(int index) { style::color fromNameFgSelected(int index) {
t_assert(index >= 0 && index < 8); Expects(index >= 0 && index < 8);
style::color colors[] = { style::color colors[] = {
st::historyPeer1NameFgSelected, st::historyPeer1NameFgSelected,
st::historyPeer2NameFgSelected, st::historyPeer2NameFgSelected,
@ -149,8 +149,10 @@ void HistoryMessageVia::resize(int32 availw) const {
} }
void HistoryMessageSigned::create(UserData *from, const QDateTime &date) { void HistoryMessageSigned::create(UserData *from, const QDateTime &date) {
QString time = qsl(", ") + date.toString(cTimeFormat()), name = App::peerName(from); auto time = qsl(", ") + date.toString(cTimeFormat());
int32 timew = st::msgDateFont->width(time), namew = st::msgDateFont->width(name); auto name = App::peerName(from);
auto timew = st::msgDateFont->width(time);
auto namew = st::msgDateFont->width(name);
if (timew + namew > st::maxSignatureSize) { if (timew + namew > st::maxSignatureSize) {
name = st::msgDateFont->elided(from->firstName, st::maxSignatureSize - timew); name = st::msgDateFont->elided(from->firstName, st::maxSignatureSize - timew);
} }
@ -164,7 +166,7 @@ int HistoryMessageSigned::maxWidth() const {
void HistoryMessageEdited::create(const QDateTime &editDate, const QDateTime &date) { void HistoryMessageEdited::create(const QDateTime &editDate, const QDateTime &date) {
_editDate = editDate; _editDate = editDate;
QString time = date.toString(cTimeFormat()); auto time = date.toString(cTimeFormat());
_edited.setText(st::msgDateTextStyle, lang(lng_edited) + ' ' + time, _textNameOptions); _edited.setText(st::msgDateTextStyle, lang(lng_edited) + ' ' + time, _textNameOptions);
} }
@ -402,7 +404,7 @@ int HistoryMessage::KeyboardStyle::minButtonWidth(HistoryMessageReplyMarkup::But
return result; return result;
} }
HistoryMessage::HistoryMessage(History *history, const MTPDmessage &msg) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, const MTPDmessage &msg)
: HistoryItem(history, msg.vid.v, msg.vflags.v, ::date(msg.vdate), msg.has_from_id() ? msg.vfrom_id.v : 0) { : HistoryItem(history, msg.vid.v, msg.vflags.v, ::date(msg.vdate), msg.has_from_id() ? msg.vfrom_id.v : 0) {
CreateConfig config; CreateConfig config;
@ -430,7 +432,7 @@ HistoryMessage::HistoryMessage(History *history, const MTPDmessage &msg)
setText({ text, entities }); setText({ text, entities });
} }
HistoryMessage::HistoryMessage(History *history, const MTPDmessageService &msg) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, const MTPDmessageService &msg)
: HistoryItem(history, msg.vid.v, mtpCastFlags(msg.vflags.v), ::date(msg.vdate), msg.has_from_id() ? msg.vfrom_id.v : 0) { : HistoryItem(history, msg.vid.v, mtpCastFlags(msg.vflags.v), ::date(msg.vdate), msg.has_from_id() ? msg.vfrom_id.v : 0) {
CreateConfig config; CreateConfig config;
@ -449,7 +451,7 @@ HistoryMessage::HistoryMessage(History *history, const MTPDmessageService &msg)
setText(TextWithEntities {}); setText(TextWithEntities {});
} }
HistoryMessage::HistoryMessage(History *history, MsgId id, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, MsgId id, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd)
: HistoryItem(history, id, newForwardedFlags(history->peer, from, fwd) | flags, date, from) { : HistoryItem(history, id, newForwardedFlags(history->peer, from, fwd) | flags, date, from) {
CreateConfig config; CreateConfig config;
@ -496,14 +498,14 @@ HistoryMessage::HistoryMessage(History *history, MsgId id, MTPDmessage::Flags fl
setText(fwd->originalText()); setText(fwd->originalText());
} }
HistoryMessage::HistoryMessage(History *history, MsgId id, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, MsgId id, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities)
: HistoryItem(history, id, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) { : HistoryItem(history, id, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) {
createComponentsHelper(flags, replyTo, viaBotId, MTPnullMarkup); createComponentsHelper(flags, replyTo, viaBotId, MTPnullMarkup);
setText(textWithEntities); setText(textWithEntities);
} }
HistoryMessage::HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup)
: HistoryItem(history, msgId, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) { : HistoryItem(history, msgId, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) {
createComponentsHelper(flags, replyTo, viaBotId, markup); createComponentsHelper(flags, replyTo, viaBotId, markup);
@ -511,7 +513,7 @@ HistoryMessage::HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags
setText(TextWithEntities()); setText(TextWithEntities());
} }
HistoryMessage::HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup)
: HistoryItem(history, msgId, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) { : HistoryItem(history, msgId, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) {
createComponentsHelper(flags, replyTo, viaBotId, markup); createComponentsHelper(flags, replyTo, viaBotId, markup);
@ -519,7 +521,7 @@ HistoryMessage::HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags
setText(TextWithEntities()); setText(TextWithEntities());
} }
HistoryMessage::HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, GameData *game, const MTPReplyMarkup &markup) HistoryMessage::HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, GameData *game, const MTPReplyMarkup &markup)
: HistoryItem(history, msgId, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) { : HistoryItem(history, msgId, flags, date, (flags & MTPDmessage::Flag::f_from_id) ? from : 0) {
createComponentsHelper(flags, replyTo, viaBotId, markup); createComponentsHelper(flags, replyTo, viaBotId, markup);
@ -899,26 +901,29 @@ bool HistoryMessage::drawBubble() const {
return _media ? (!emptyText() || _media->needsBubble()) : !isEmpty(); return _media ? (!emptyText() || _media->needsBubble()) : !isEmpty();
} }
void HistoryMessage::countPositionAndSize(int32 &left, int32 &width) const { QRect HistoryMessage::countGeometry() const {
int32 maxwidth = qMin(int(st::msgMaxWidth), _maxw), hwidth = _history->width; auto maxwidth = qMin(st::msgMaxWidth, _maxw);
if (_media && _media->currentWidth() < maxwidth) { if (_media && _media->currentWidth() < maxwidth) {
maxwidth = qMax(_media->currentWidth(), qMin(maxwidth, plainMaxWidth())); maxwidth = qMax(_media->currentWidth(), qMin(maxwidth, plainMaxWidth()));
} }
left = (!isPost() && out() && !Adaptive::ChatWide()) ? st::msgMargin.right() : st::msgMargin.left(); auto contentLeft = (!isPost() && out() && !Adaptive::ChatWide()) ? st::msgMargin.right() : st::msgMargin.left();
if (hasFromPhoto()) { if (hasFromPhoto()) {
left += st::msgPhotoSkip; contentLeft += st::msgPhotoSkip;
// } else if (!Adaptive::Wide() && !out() && !fromChannel() && st::msgPhotoSkip - (hmaxwidth - hwidth) > 0) { // } else if (!Adaptive::Wide() && !out() && !fromChannel() && st::msgPhotoSkip - (hmaxwidth - hwidth) > 0) {
// left += st::msgPhotoSkip - (hmaxwidth - hwidth); // contentLeft += st::msgPhotoSkip - (hmaxwidth - hwidth);
} }
width = hwidth - st::msgMargin.left() - st::msgMargin.right(); auto contentWidth = width() - st::msgMargin.left() - st::msgMargin.right();
if (width > maxwidth) { if (contentWidth > maxwidth) {
if (!isPost() && out() && !Adaptive::ChatWide()) { if (!isPost() && out() && !Adaptive::ChatWide()) {
left += width - maxwidth; contentLeft += contentWidth - maxwidth;
} }
width = maxwidth; contentWidth = maxwidth;
} }
auto contentTop = marginTop();
return QRect(contentLeft, contentTop, contentWidth, _height - contentTop - marginBottom());
} }
void HistoryMessage::fromNameUpdated(int32 width) const { void HistoryMessage::fromNameUpdated(int32 width) const {
@ -1317,22 +1322,23 @@ void HistoryMessage::setId(MsgId newId) {
} }
} }
void HistoryMessage::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { void HistoryMessage::draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const {
bool outbg = out() && !isPost(), bubble = drawBubble(), selected = (selection == FullSelection); bool outbg = out() && !isPost(), bubble = drawBubble(), selected = (selection == FullSelection);
int left = 0, width = 0, height = _height; auto g = countGeometry();
countPositionAndSize(left, width); if (g.width() < 1) {
if (width < 1) return; return;
}
int dateh = 0, unreadbarh = 0; auto dateh = 0;
if (auto date = Get<HistoryMessageDate>()) { if (auto date = Get<HistoryMessageDate>()) {
dateh = date->height(); dateh = date->height();
} }
if (auto unreadbar = Get<HistoryMessageUnreadBar>()) { if (auto unreadbar = Get<HistoryMessageUnreadBar>()) {
unreadbarh = unreadbar->height(); auto unreadbarh = unreadbar->height();
if (r.intersects(QRect(0, dateh, _history->width, unreadbarh))) { if (clip.intersects(QRect(0, dateh, width(), unreadbarh))) {
p.translate(0, dateh); p.translate(0, dateh);
unreadbar->paint(p, 0, _history->width); unreadbar->paint(p, 0, width());
p.translate(0, -dateh); p.translate(0, -dateh);
} }
} }
@ -1347,12 +1353,12 @@ void HistoryMessage::draw(Painter &p, const QRect &r, TextSelection selection, T
auto bottom = marginBottom(); auto bottom = marginBottom();
auto fill = qMin(top, bottom); auto fill = qMin(top, bottom);
auto skiptop = top - fill; auto skiptop = top - fill;
auto fillheight = fill + (height - top - bottom) + fill; auto fillheight = fill + g.height() + fill;
auto dt = (animms > st::activeFadeInDuration) ? (1. - (animms - st::activeFadeInDuration) / float64(st::activeFadeOutDuration)) : (animms / float64(st::activeFadeInDuration)); auto dt = (animms > st::activeFadeInDuration) ? (1. - (animms - st::activeFadeInDuration) / float64(st::activeFadeOutDuration)) : (animms / float64(st::activeFadeInDuration));
auto o = p.opacity(); auto o = p.opacity();
p.setOpacity(o * dt); p.setOpacity(o * dt);
p.fillRect(0, skiptop, _history->width, fillheight, st::defaultTextPalette.selectOverlay); p.fillRect(0, skiptop, width(), fillheight, st::defaultTextPalette.selectOverlay);
p.setOpacity(o); p.setOpacity(o);
} }
} }
@ -1361,29 +1367,27 @@ void HistoryMessage::draw(Painter &p, const QRect &r, TextSelection selection, T
auto keyboard = inlineReplyKeyboard(); auto keyboard = inlineReplyKeyboard();
if (keyboard) { if (keyboard) {
int h = st::msgBotKbButton.margin + keyboard->naturalHeight(); auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight();
height -= h; g.setHeight(g.height() - keyboardHeight);
int top = height + st::msgBotKbButton.margin - marginBottom(); auto keyboardPosition = QPoint(g.left(), g.top() + g.height() + st::msgBotKbButton.margin);
p.translate(left, top); p.translate(keyboardPosition);
keyboard->paint(p, width, r.translated(-left, -top), ms); keyboard->paint(p, g.width(), clip.translated(-keyboardPosition), ms);
p.translate(-left, -top); p.translate(-keyboardPosition);
} }
if (bubble) { if (bubble) {
if (displayFromName() && author()->nameVersion > _authorNameVersion) { if (displayFromName() && author()->nameVersion > _authorNameVersion) {
fromNameUpdated(width); fromNameUpdated(g.width());
} }
auto entry = Get<HistoryMessageLogEntryOriginal>(); auto entry = Get<HistoryMessageLogEntryOriginal>();
auto mediaDisplayed = _media && _media->isDisplayed(); auto mediaDisplayed = _media && _media->isDisplayed();
auto top = marginTop();
auto r = QRect(left, top, width, height - top - marginBottom());
auto skipTail = isAttachedToNext() || (_media && _media->skipBubbleTail()) || (keyboard != nullptr); auto skipTail = isAttachedToNext() || (_media && _media->skipBubbleTail()) || (keyboard != nullptr);
auto displayTail = skipTail ? RectPart::None : (outbg && !Adaptive::ChatWide()) ? RectPart::Right : RectPart::Left; auto displayTail = skipTail ? RectPart::None : (outbg && !Adaptive::ChatWide()) ? RectPart::Right : RectPart::Left;
HistoryLayout::paintBubble(p, r, _history->width, selected, outbg, displayTail); HistoryLayout::paintBubble(p, g, width(), selected, outbg, displayTail);
auto trect = r.marginsAdded(-st::msgPadding); auto trect = g.marginsAdded(-st::msgPadding);
if (mediaDisplayed && _media->isBubbleTop()) { if (mediaDisplayed && _media->isBubbleTop()) {
trect.setY(trect.y() - st::msgPadding.top()); trect.setY(trect.y() - st::msgPadding.top());
} else { } else {
@ -1402,13 +1406,13 @@ void HistoryMessage::draw(Painter &p, const QRect &r, TextSelection selection, T
if (mediaDisplayed) { if (mediaDisplayed) {
auto mediaAboveText = _media->isAboveMessage(); auto mediaAboveText = _media->isAboveMessage();
auto mediaHeight = _media->height(); auto mediaHeight = _media->height();
auto mediaLeft = trect.x() - st::msgPadding.left(); auto mediaLeft = g.left();
auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight); auto mediaTop = mediaAboveText ? trect.y() : (trect.y() + trect.height() - mediaHeight);
if (!mediaAboveText) { if (!mediaAboveText) {
paintText(p, trect, selection); paintText(p, trect, selection);
} }
p.translate(mediaLeft, mediaTop); p.translate(mediaLeft, mediaTop);
_media->draw(p, r.translated(-mediaLeft, -mediaTop), toMediaSelection(selection), ms); _media->draw(p, clip.translated(-mediaLeft, -mediaTop), toMediaSelection(selection), ms);
p.translate(-mediaLeft, -mediaTop); p.translate(-mediaLeft, -mediaTop);
if (mediaAboveText) { if (mediaAboveText) {
@ -1421,20 +1425,19 @@ void HistoryMessage::draw(Painter &p, const QRect &r, TextSelection selection, T
paintText(p, trect, selection); paintText(p, trect, selection);
} }
if (entry) { if (entry) {
auto entryLeft = r.x(); auto entryLeft = g.left();
auto entryTop = trect.y() + trect.height(); auto entryTop = trect.y() + trect.height();
p.translate(entryLeft, entryTop); p.translate(entryLeft, entryTop);
entry->_page->draw(p, r.translated(-entryLeft, -entryTop), TextSelection(), ms); entry->_page->draw(p, clip.translated(-entryLeft, -entryTop), TextSelection(), ms);
p.translate(-entryLeft, -entryTop); p.translate(-entryLeft, -entryTop);
} }
if (needDrawInfo) { if (needDrawInfo) {
HistoryMessage::drawInfo(p, r.x() + r.width(), r.y() + r.height(), 2 * r.x() + r.width(), selected, InfoDisplayDefault); HistoryMessage::drawInfo(p, g.left() + g.width(), g.top() + g.height(), 2 * g.left() + g.width(), selected, InfoDisplayDefault);
} }
} else if (_media) { } else if (_media) {
auto top = marginTop(); p.translate(g.topLeft());
p.translate(left, top); _media->draw(p, clip.translated(-g.topLeft()), toMediaSelection(selection), ms);
_media->draw(p, r.translated(-left, -top), toMediaSelection(selection), ms); p.translate(-g.topLeft());
p.translate(-left, -top);
} }
p.restoreTextPalette(); p.restoreTextPalette();
@ -1503,7 +1506,7 @@ void HistoryMessage::paintViaBotIdInfo(Painter &p, QRect &trect, bool selected)
if (auto via = Get<HistoryMessageVia>()) { if (auto via = Get<HistoryMessageVia>()) {
p.setFont(st::msgServiceNameFont); p.setFont(st::msgServiceNameFont);
p.setPen(selected ? (hasOutLayout() ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (hasOutLayout() ? st::msgOutServiceFg : st::msgInServiceFg)); p.setPen(selected ? (hasOutLayout() ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (hasOutLayout() ? st::msgOutServiceFg : st::msgInServiceFg));
p.drawTextLeft(trect.left(), trect.top(), _history->width, via->_text); p.drawTextLeft(trect.left(), trect.top(), width(), via->_text);
trect.setY(trect.y() + st::msgServiceNameFont->height); trect.setY(trect.y() + st::msgServiceNameFont->height);
} }
} }
@ -1523,8 +1526,8 @@ void HistoryMessage::dependencyItemRemoved(HistoryItem *dependency) {
} }
} }
int HistoryMessage::resizeGetHeight_(int width) { int HistoryMessage::resizeContentGetHeight() {
int result = performResizeGetHeight(width); int result = performResizeGetHeight();
auto keyboard = inlineReplyKeyboard(); auto keyboard = inlineReplyKeyboard();
if (auto markup = Get<HistoryMessageReplyMarkup>()) { if (auto markup = Get<HistoryMessageReplyMarkup>()) {
@ -1544,14 +1547,14 @@ int HistoryMessage::resizeGetHeight_(int width) {
return result; return result;
} }
int HistoryMessage::performResizeGetHeight(int width) { int HistoryMessage::performResizeGetHeight() {
if (width < st::msgMinWidth) return _height; if (width() < st::msgMinWidth) return _height;
width -= st::msgMargin.left() + st::msgMargin.right(); auto contentWidth = width() - (st::msgMargin.left() + st::msgMargin.right());
if (width < st::msgPadding.left() + st::msgPadding.right() + 1) { if (contentWidth < st::msgPadding.left() + st::msgPadding.right() + 1) {
width = st::msgPadding.left() + st::msgPadding.right() + 1; contentWidth = st::msgPadding.left() + st::msgPadding.right() + 1;
} else if (width > st::msgMaxWidth) { } else if (contentWidth > st::msgMaxWidth) {
width = st::msgMaxWidth; contentWidth = st::msgMaxWidth;
} }
if (drawBubble()) { if (drawBubble()) {
auto forwarded = Get<HistoryMessageForwarded>(); auto forwarded = Get<HistoryMessageForwarded>();
@ -1565,7 +1568,7 @@ int HistoryMessage::performResizeGetHeight(int width) {
mediaDisplayed = _media->isDisplayed(); mediaDisplayed = _media->isDisplayed();
mediaInBubbleState = _media->inBubbleState(); mediaInBubbleState = _media->inBubbleState();
} }
if (width >= _maxw) { if (contentWidth >= _maxw) {
_height = _minh; _height = _minh;
if (mediaDisplayed) { if (mediaDisplayed) {
_media->resizeGetHeight(_maxw); _media->resizeGetHeight(_maxw);
@ -1577,7 +1580,7 @@ int HistoryMessage::performResizeGetHeight(int width) {
if (emptyText()) { if (emptyText()) {
_height = 0; _height = 0;
} else { } else {
auto textWidth = qMax(width - st::msgPadding.left() - st::msgPadding.right(), 1); auto textWidth = qMax(contentWidth - st::msgPadding.left() - st::msgPadding.right(), 1);
if (textWidth != _textWidth) { if (textWidth != _textWidth) {
_textWidth = textWidth; _textWidth = textWidth;
_textHeight = _text.countHeight(textWidth); _textHeight = _text.countHeight(textWidth);
@ -1591,52 +1594,46 @@ int HistoryMessage::performResizeGetHeight(int width) {
if (!_media->isBubbleBottom()) { if (!_media->isBubbleBottom()) {
_height += st::msgPadding.bottom() + st::mediaInBubbleSkip; _height += st::msgPadding.bottom() + st::mediaInBubbleSkip;
} }
_height += _media->resizeGetHeight(width); _height += _media->resizeGetHeight(contentWidth);
} else { } else {
_height += st::msgPadding.top() + st::msgPadding.bottom(); _height += st::msgPadding.top() + st::msgPadding.bottom();
} }
if (entry) { if (entry) {
_height += entry->_page->resizeGetHeight(width); _height += entry->_page->resizeGetHeight(contentWidth);
} }
} }
if (displayFromName()) { if (displayFromName()) {
int32 l = 0, w = 0; auto g = countGeometry();
countPositionAndSize(l, w); fromNameUpdated(g.width());
fromNameUpdated(w);
_height += st::msgNameFont->height; _height += st::msgNameFont->height;
} else if (via && !forwarded) { } else if (via && !forwarded) {
int32 l = 0, w = 0; auto g = countGeometry();
countPositionAndSize(l, w); via->resize(g.width() - st::msgPadding.left() - st::msgPadding.right());
via->resize(w - st::msgPadding.left() - st::msgPadding.right());
_height += st::msgNameFont->height; _height += st::msgNameFont->height;
} }
if (displayForwardedFrom()) { if (displayForwardedFrom()) {
int32 l = 0, w = 0; auto g = countGeometry();
countPositionAndSize(l, w); auto fwdheight = ((forwarded->_text.maxWidth() > (g.width() - st::msgPadding.left() - st::msgPadding.right())) ? 2 : 1) * st::semiboldFont->height;
int32 fwdheight = ((forwarded->_text.maxWidth() > (w - st::msgPadding.left() - st::msgPadding.right())) ? 2 : 1) * st::semiboldFont->height;
_height += fwdheight; _height += fwdheight;
} }
if (reply) { if (reply) {
int32 l = 0, w = 0; auto g = countGeometry();
countPositionAndSize(l, w); reply->resize(g.width() - st::msgPadding.left() - st::msgPadding.right());
reply->resize(w - st::msgPadding.left() - st::msgPadding.right());
_height += st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); _height += st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom();
} }
} else if (_media) { } else if (_media) {
_height = _media->resizeGetHeight(width); _height = _media->resizeGetHeight(contentWidth);
} else { } else {
_height = 0; _height = 0;
} }
if (auto keyboard = inlineReplyKeyboard()) { if (auto keyboard = inlineReplyKeyboard()) {
int32 l = 0, w = 0; auto g = countGeometry();
countPositionAndSize(l, w); auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight();
_height += keyboardHeight;
int h = st::msgBotKbButton.margin + keyboard->naturalHeight(); keyboard->resize(g.width(), keyboardHeight - st::msgBotKbButton.margin);
_height += h;
keyboard->resize(w, h - st::msgBotKbButton.margin);
} }
_height += marginTop() + marginBottom(); _height += marginTop() + marginBottom();
@ -1644,16 +1641,15 @@ int HistoryMessage::performResizeGetHeight(int width) {
} }
bool HistoryMessage::hasPoint(int x, int y) const { bool HistoryMessage::hasPoint(int x, int y) const {
int left = 0, width = 0, height = _height; auto g = countGeometry();
countPositionAndSize(left, width); if (g.width() < 1) {
if (width < 1) return false; return false;
}
if (drawBubble()) { if (drawBubble()) {
int top = marginTop(); return g.contains(x, y);
QRect r(left, top, width, height - top - marginBottom());
return r.contains(x, y);
} else if (_media) { } else if (_media) {
return _media->hasPoint(x - left, y - marginTop()); return _media->hasPoint(x - g.left(), y - g.top());
} else { } else {
return false; return false;
} }
@ -1679,22 +1675,21 @@ bool HistoryMessage::pointInTime(int32 right, int32 bottom, int x, int y, InfoDi
HistoryTextState HistoryMessage::getState(int x, int y, HistoryStateRequest request) const { HistoryTextState HistoryMessage::getState(int x, int y, HistoryStateRequest request) const {
HistoryTextState result; HistoryTextState result;
int left = 0, width = 0, height = _height; auto g = countGeometry();
countPositionAndSize(left, width); if (g.width() < 1) {
return result;
if (width < 1) return result; }
auto keyboard = inlineReplyKeyboard(); auto keyboard = inlineReplyKeyboard();
auto keyboardHeight = 0;
if (keyboard) { if (keyboard) {
int h = st::msgBotKbButton.margin + keyboard->naturalHeight(); keyboardHeight = keyboard->naturalHeight();
height -= h; g.setHeight(g.height() - st::msgBotKbButton.margin - keyboardHeight);
} }
if (drawBubble()) { if (drawBubble()) {
auto mediaDisplayed = _media && _media->isDisplayed(); auto mediaDisplayed = _media && _media->isDisplayed();
auto top = marginTop(); auto trect = g.marginsAdded(-st::msgPadding);
QRect r(left, top, width, height - top - marginBottom());
QRect trect(r.marginsAdded(-st::msgPadding));
if (mediaDisplayed && _media->isBubbleTop()) { if (mediaDisplayed && _media->isBubbleTop()) {
trect.setY(trect.y() - st::msgPadding.top()); trect.setY(trect.y() - st::msgPadding.top());
} else { } else {
@ -1729,19 +1724,19 @@ HistoryTextState HistoryMessage::getState(int x, int y, HistoryStateRequest requ
getStateText(x, y, trect, &result, request); getStateText(x, y, trect, &result, request);
} }
if (needDateCheck) { if (needDateCheck) {
if (HistoryMessage::pointInTime(r.x() + r.width(), r.y() + r.height(), x, y, InfoDisplayDefault)) { if (HistoryMessage::pointInTime(g.left() + g.width(), g.top() + g.height(), x, y, InfoDisplayDefault)) {
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
} }
} else if (_media) { } else if (_media) {
result = _media->getState(x - left, y - marginTop(), request); result = _media->getState(x - g.left(), y - g.top(), request);
result.symbol += _text.length(); result.symbol += _text.length();
} }
if (keyboard) { if (keyboard) {
int top = height + st::msgBotKbButton.margin - marginBottom(); auto keyboardTop = g.top() + g.height() + st::msgBotKbButton.margin;
if (x >= left && x < left + width && y >= top && y < _height - marginBottom()) { if (QRect(g.left(), keyboardTop, g.width(), keyboardHeight).contains(x, y)) {
result.link = keyboard->getState(x - left, y - top); result.link = keyboard->getState(x - g.left(), y - keyboardTop);
return result; return result;
} }
} }
@ -1753,20 +1748,17 @@ HistoryTextState HistoryMessage::getState(int x, int y, HistoryStateRequest requ
void HistoryMessage::updatePressed(int x, int y) { void HistoryMessage::updatePressed(int x, int y) {
if (!_media) return; if (!_media) return;
auto left = 0, width = 0, height = _height; auto g = countGeometry();
countPositionAndSize(left, width);
auto keyboard = inlineReplyKeyboard(); auto keyboard = inlineReplyKeyboard();
if (keyboard) { if (keyboard) {
auto h = st::msgBotKbButton.margin + keyboard->naturalHeight(); auto keyboardHeight = st::msgBotKbButton.margin + keyboard->naturalHeight();
height -= h; g.setHeight(g.height() - keyboardHeight);
} }
if (drawBubble()) { if (drawBubble()) {
auto mediaDisplayed = _media && _media->isDisplayed(); auto mediaDisplayed = _media && _media->isDisplayed();
auto top = marginTop(); auto top = marginTop();
QRect r(left, top, width, height - top - marginBottom()); auto trect = g.marginsAdded(-st::msgPadding);
QRect trect(r.marginsAdded(-st::msgPadding));
if (mediaDisplayed && _media->isBubbleTop()) { if (mediaDisplayed && _media->isBubbleTop()) {
trect.setY(trect.y() - st::msgPadding.top()); trect.setY(trect.y() - st::msgPadding.top());
} else { } else {
@ -1799,7 +1791,7 @@ void HistoryMessage::updatePressed(int x, int y) {
_media->updatePressed(x - mediaLeft, y - mediaTop); _media->updatePressed(x - mediaLeft, y - mediaTop);
} }
} else { } else {
_media->updatePressed(x - left, y - marginTop()); _media->updatePressed(x - g.left(), y - g.top());
} }
} }
@ -2252,13 +2244,13 @@ HistoryService::PreparedText HistoryService::preparePaymentSentText() {
return result; return result;
} }
HistoryService::HistoryService(History *history, const MTPDmessageService &message) : HistoryService::HistoryService(gsl::not_null<History*> history, const MTPDmessageService &message) :
HistoryItem(history, message.vid.v, mtpCastFlags(message.vflags.v), ::date(message.vdate), message.has_from_id() ? message.vfrom_id.v : 0) { HistoryItem(history, message.vid.v, mtpCastFlags(message.vflags.v), ::date(message.vdate), message.has_from_id() ? message.vfrom_id.v : 0) {
createFromMtp(message); createFromMtp(message);
setMessageByAction(message.vaction); setMessageByAction(message.vaction);
} }
HistoryService::HistoryService(History *history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags, int32 from, PhotoData *photo) : HistoryService::HistoryService(gsl::not_null<History*> history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags, int32 from, PhotoData *photo) :
HistoryItem(history, msgId, flags, date, from) { HistoryItem(history, msgId, flags, date, from) {
setServiceText(message); setServiceText(message);
if (photo) { if (photo) {
@ -2279,13 +2271,12 @@ bool HistoryService::updateDependencyItem() {
return HistoryItem::updateDependencyItem(); return HistoryItem::updateDependencyItem();
} }
void HistoryService::countPositionAndSize(int32 &left, int32 &width) const { QRect HistoryService::countGeometry() const {
left = st::msgServiceMargin.left(); auto result = QRect(0, 0, width(), _height);
int32 maxwidth = _history->width;
if (Adaptive::ChatWide()) { if (Adaptive::ChatWide()) {
maxwidth = qMin(maxwidth, int32(st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left())); result.setWidth(qMin(result.width(), st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left()));
} }
width = maxwidth - st::msgServiceMargin.left() - st::msgServiceMargin.left(); return result.marginsRemoved(st::msgServiceMargin);
} }
TextWithEntities HistoryService::selectedText(TextSelection selection) const { TextWithEntities HistoryService::selectedText(TextSelection selection) const {
@ -2314,24 +2305,20 @@ void HistoryService::setServiceText(const PreparedText &prepared) {
_textHeight = 0; _textHeight = 0;
} }
void HistoryService::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const { void HistoryService::draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const {
int height = _height - st::msgServiceMargin.top() - st::msgServiceMargin.bottom(); auto height = _height - st::msgServiceMargin.top() - st::msgServiceMargin.bottom();
auto dateh = 0;
QRect clip(r); auto unreadbarh = 0;
int dateh = 0, unreadbarh = 0;
if (auto date = Get<HistoryMessageDate>()) { if (auto date = Get<HistoryMessageDate>()) {
dateh = date->height(); dateh = date->height();
//if (clip.intersects(QRect(0, 0, _history->width, dateh))) {
// date->paint(p, 0, _history->width);
//}
p.translate(0, dateh); p.translate(0, dateh);
clip.translate(0, -dateh); clip.translate(0, -dateh);
height -= dateh; height -= dateh;
} }
if (auto unreadbar = Get<HistoryMessageUnreadBar>()) { if (auto unreadbar = Get<HistoryMessageUnreadBar>()) {
unreadbarh = unreadbar->height(); unreadbarh = unreadbar->height();
if (clip.intersects(QRect(0, 0, _history->width, unreadbarh))) { if (clip.intersects(QRect(0, 0, width(), unreadbarh))) {
unreadbar->paint(p, 0, _history->width); unreadbar->paint(p, 0, width());
} }
p.translate(0, unreadbarh); p.translate(0, unreadbarh);
clip.translate(0, -unreadbarh); clip.translate(0, -unreadbarh);
@ -2341,12 +2328,12 @@ void HistoryService::draw(Painter &p, const QRect &r, TextSelection selection, T
HistoryLayout::PaintContext context(ms, clip, selection); HistoryLayout::PaintContext context(ms, clip, selection);
HistoryLayout::ServiceMessagePainter::paint(p, this, context, height); HistoryLayout::ServiceMessagePainter::paint(p, this, context, height);
if (int skiph = dateh + unreadbarh) { if (auto skiph = dateh + unreadbarh) {
p.translate(0, -skiph); p.translate(0, -skiph);
} }
} }
int32 HistoryService::resizeGetHeight_(int32 width) { int HistoryService::resizeContentGetHeight() {
_height = displayedDateHeight(); _height = displayedDateHeight();
if (auto unreadbar = Get<HistoryMessageUnreadBar>()) { if (auto unreadbar = Get<HistoryMessageUnreadBar>()) {
_height += unreadbar->height(); _height += unreadbar->height();
@ -2355,20 +2342,21 @@ int32 HistoryService::resizeGetHeight_(int32 width) {
if (_text.isEmpty()) { if (_text.isEmpty()) {
_textHeight = 0; _textHeight = 0;
} else { } else {
int32 maxwidth = _history->width; auto contentWidth = width();
if (Adaptive::ChatWide()) { if (Adaptive::ChatWide()) {
maxwidth = qMin(maxwidth, int32(st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left())); accumulate_min(contentWidth, st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left());
}
contentWidth -= st::msgServiceMargin.left() + st::msgServiceMargin.left(); // two small margins
if (contentWidth < st::msgServicePadding.left() + st::msgServicePadding.right() + 1) {
contentWidth = st::msgServicePadding.left() + st::msgServicePadding.right() + 1;
} }
if (width > maxwidth) width = maxwidth;
width -= st::msgServiceMargin.left() + st::msgServiceMargin.left(); // two small margins
if (width < st::msgServicePadding.left() + st::msgServicePadding.right() + 1) width = st::msgServicePadding.left() + st::msgServicePadding.right() + 1;
int32 nwidth = qMax(width - st::msgServicePadding.left() - st::msgServicePadding.right(), 0); auto nwidth = qMax(contentWidth - st::msgServicePadding.left() - st::msgServicePadding.right(), 0);
if (nwidth != _textWidth) { if (nwidth != _textWidth) {
_textWidth = nwidth; _textWidth = nwidth;
_textHeight = _text.countHeight(nwidth); _textHeight = _text.countHeight(nwidth);
} }
if (width >= _maxw) { if (contentWidth >= _maxw) {
_height += _minh; _height += _minh;
} else { } else {
_height += _textHeight; _height += _textHeight;
@ -2383,63 +2371,60 @@ int32 HistoryService::resizeGetHeight_(int32 width) {
} }
bool HistoryService::hasPoint(int x, int y) const { bool HistoryService::hasPoint(int x, int y) const {
int left = 0, width = 0, height = _height - st::msgServiceMargin.top() - st::msgServiceMargin.bottom(); // two small margins auto g = countGeometry();
countPositionAndSize(left, width); if (g.width() < 1) {
if (width < 1) return false; return false;
}
if (int dateh = displayedDateHeight()) { if (auto dateh = displayedDateHeight()) {
y -= dateh; g.setTop(g.top() + dateh);
height -= dateh;
} }
if (auto unreadbar = Get<HistoryMessageUnreadBar>()) { if (auto unreadbar = Get<HistoryMessageUnreadBar>()) {
int unreadbarh = unreadbar->height(); g.setTop(g.top() + unreadbar->height());
y -= unreadbarh;
height -= unreadbarh;
} }
if (_media) { if (_media) {
height -= st::msgServiceMargin.top() + _media->height(); g.setHeight(g.height() - (st::msgServiceMargin.top() + _media->height()));
} }
return QRect(left, st::msgServiceMargin.top(), width, height).contains(x, y); return g.contains(x, y);
} }
HistoryTextState HistoryService::getState(int x, int y, HistoryStateRequest request) const { HistoryTextState HistoryService::getState(int x, int y, HistoryStateRequest request) const {
HistoryTextState result; HistoryTextState result;
int left = 0, width = 0, height = _height - st::msgServiceMargin.top() - st::msgServiceMargin.bottom(); // two small margins auto g = countGeometry();
countPositionAndSize(left, width); if (g.width() < 1) {
if (width < 1) return result; return result;
}
if (int dateh = displayedDateHeight()) { if (auto dateh = displayedDateHeight()) {
y -= dateh; y -= dateh;
height -= dateh; g.setHeight(g.height() - dateh);
} }
if (auto unreadbar = Get<HistoryMessageUnreadBar>()) { if (auto unreadbar = Get<HistoryMessageUnreadBar>()) {
int unreadbarh = unreadbar->height(); auto unreadbarh = unreadbar->height();
y -= unreadbarh; y -= unreadbarh;
height -= unreadbarh; g.setHeight(g.height() - unreadbarh);
} }
if (_media) { if (_media) {
height -= st::msgServiceMargin.top() + _media->height(); g.setHeight(g.height() - (st::msgServiceMargin.top() + _media->height()));
} }
auto outer = QRect(left, st::msgServiceMargin.top(), width, height); auto trect = g.marginsAdded(-st::msgServicePadding);
auto trect = outer.marginsAdded(-st::msgServicePadding);
if (trect.contains(x, y)) { if (trect.contains(x, y)) {
auto textRequest = request.forText(); auto textRequest = request.forText();
textRequest.align = style::al_center; textRequest.align = style::al_center;
result = _text.getState(x - trect.x(), y - trect.y(), trect.width(), textRequest); result = _text.getState(x - trect.x(), y - trect.y(), trect.width(), textRequest);
if (auto gamescore = Get<HistoryServiceGameScore>()) { if (auto gamescore = Get<HistoryServiceGameScore>()) {
if (!result.link && result.cursor == HistoryInTextCursorState && outer.contains(x, y)) { if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(x, y)) {
result.link = gamescore->lnk; result.link = gamescore->lnk;
} }
} else if (auto payment = Get<HistoryServicePayment>()) { } else if (auto payment = Get<HistoryServicePayment>()) {
if (!result.link && result.cursor == HistoryInTextCursorState && outer.contains(x, y)) { if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(x, y)) {
result.link = payment->lnk; result.link = payment->lnk;
} }
} }
} else if (_media) { } else if (_media) {
result = _media->getState(x - st::msgServiceMargin.left() - (width - _media->maxWidth()) / 2, y - st::msgServiceMargin.top() - height - st::msgServiceMargin.top(), request); result = _media->getState(x - st::msgServiceMargin.left() - (g.width() - _media->maxWidth()) / 2, y - st::msgServiceMargin.top() - g.height() - st::msgServiceMargin.top(), request);
} }
return result; return result;
} }

View File

@ -24,25 +24,25 @@ void historyInitMessages();
class HistoryMessage : public HistoryItem, private HistoryItemInstantiated<HistoryMessage> { class HistoryMessage : public HistoryItem, private HistoryItemInstantiated<HistoryMessage> {
public: public:
static HistoryMessage *create(History *history, const MTPDmessage &msg) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, const MTPDmessage &msg) {
return _create(history, msg); return _create(history, msg);
} }
static HistoryMessage *create(History *history, const MTPDmessageService &msg) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, const MTPDmessageService &msg) {
return _create(history, msg); return _create(history, msg);
} }
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd) {
return _create(history, msgId, flags, date, from, fwd); return _create(history, msgId, flags, date, from, fwd);
} }
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities) {
return _create(history, msgId, flags, replyTo, viaBotId, date, from, textWithEntities); return _create(history, msgId, flags, replyTo, viaBotId, date, from, textWithEntities);
} }
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup) {
return _create(history, msgId, flags, replyTo, viaBotId, date, from, doc, caption, markup); return _create(history, msgId, flags, replyTo, viaBotId, date, from, doc, caption, markup);
} }
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup) {
return _create(history, msgId, flags, replyTo, viaBotId, date, from, photo, caption, markup); return _create(history, msgId, flags, replyTo, viaBotId, date, from, photo, caption, markup);
} }
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, GameData *game, const MTPReplyMarkup &markup) { static gsl::not_null<HistoryMessage*> create(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, GameData *game, const MTPReplyMarkup &markup) {
return _create(history, msgId, flags, replyTo, viaBotId, date, from, game, markup); return _create(history, msgId, flags, replyTo, viaBotId, date, from, game, markup);
} }
@ -52,7 +52,7 @@ public:
void fromNameUpdated(int32 width) const; void fromNameUpdated(int32 width) const;
int32 plainMaxWidth() const; int32 plainMaxWidth() const;
void countPositionAndSize(int32 &left, int32 &width) const; QRect countGeometry() const;
bool drawBubble() const; bool drawBubble() const;
bool hasBubble() const override { bool hasBubble() const override {
@ -71,7 +71,7 @@ public:
void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const override; void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const override;
void setViewsCount(int32 count) override; void setViewsCount(int32 count) override;
void setId(MsgId newId) override; void setId(MsgId newId) override;
void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const override;
void dependencyItemRemoved(HistoryItem *dependency) override; void dependencyItemRemoved(HistoryItem *dependency) override;
@ -147,13 +147,13 @@ public:
~HistoryMessage(); ~HistoryMessage();
private: private:
HistoryMessage(History *history, const MTPDmessage &msg); HistoryMessage(gsl::not_null<History*> history, const MTPDmessage &msg);
HistoryMessage(History *history, const MTPDmessageService &msg); HistoryMessage(gsl::not_null<History*> history, const MTPDmessageService &msg);
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd); // local forwarded HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd); // local forwarded
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities); // local message HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities); // local message
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup); // local document HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup); // local document
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup); // local photo HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup); // local photo
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, GameData *game, const MTPReplyMarkup &markup); // local game HistoryMessage(gsl::not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, GameData *game, const MTPReplyMarkup &markup); // local game
friend class HistoryItemInstantiated<HistoryMessage>; friend class HistoryItemInstantiated<HistoryMessage>;
void setEmptyText(); void setEmptyText();
@ -163,8 +163,8 @@ private:
void replaceBuyWithReceiptInMarkup(); void replaceBuyWithReceiptInMarkup();
void initDimensions() override; void initDimensions() override;
int resizeGetHeight_(int width) override; int resizeContentGetHeight() override;
int performResizeGetHeight(int width); int performResizeGetHeight();
void applyEditionToEmpty(); void applyEditionToEmpty();
bool displayForwardedFrom() const { bool displayForwardedFrom() const {
@ -276,10 +276,10 @@ public:
QList<ClickHandlerPtr> links; QList<ClickHandlerPtr> links;
}; };
static HistoryService *create(History *history, const MTPDmessageService &message) { static gsl::not_null<HistoryService*> create(gsl::not_null<History*> history, const MTPDmessageService &message) {
return _create(history, message); return _create(history, message);
} }
static HistoryService *create(History *history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags = 0, UserId from = 0, PhotoData *photo = nullptr) { static gsl::not_null<HistoryService*> create(gsl::not_null<History*> history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags = 0, UserId from = 0, PhotoData *photo = nullptr) {
return _create(history, msgId, date, message, flags, from, photo); return _create(history, msgId, date, message, flags, from, photo);
} }
@ -297,9 +297,9 @@ public:
return true; return true;
} }
void countPositionAndSize(int32 &left, int32 &width) const; QRect countGeometry() const;
void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override; void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const override;
bool hasPoint(int x, int y) const override; bool hasPoint(int x, int y) const override;
HistoryTextState getState(int x, int y, HistoryStateRequest request) const override; HistoryTextState getState(int x, int y, HistoryStateRequest request) const override;
@ -336,12 +336,12 @@ public:
protected: protected:
friend class HistoryLayout::ServiceMessagePainter; friend class HistoryLayout::ServiceMessagePainter;
HistoryService(History *history, const MTPDmessageService &message); HistoryService(gsl::not_null<History*> history, const MTPDmessageService &message);
HistoryService(History *history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags = 0, UserId from = 0, PhotoData *photo = 0); HistoryService(gsl::not_null<History*> history, MsgId msgId, QDateTime date, const PreparedText &message, MTPDmessage::Flags flags = 0, UserId from = 0, PhotoData *photo = 0);
friend class HistoryItemInstantiated<HistoryService>; friend class HistoryItemInstantiated<HistoryService>;
void initDimensions() override; void initDimensions() override;
int resizeGetHeight_(int width) override; int resizeContentGetHeight() override;
void setServiceText(const PreparedText &prepared); void setServiceText(const PreparedText &prepared);
@ -383,7 +383,7 @@ private:
class HistoryJoined : public HistoryService, private HistoryItemInstantiated<HistoryJoined> { class HistoryJoined : public HistoryService, private HistoryItemInstantiated<HistoryJoined> {
public: public:
static HistoryJoined *create(gsl::not_null<History*> history, const QDateTime &inviteDate, gsl::not_null<UserData*> inviter, MTPDmessage::Flags flags) { static gsl::not_null<HistoryJoined*> create(gsl::not_null<History*> history, const QDateTime &inviteDate, gsl::not_null<UserData*> inviter, MTPDmessage::Flags flags) {
return _create(history, inviteDate, inviter, flags); return _create(history, inviteDate, inviter, flags);
} }

View File

@ -187,9 +187,8 @@ int WideChatWidth() {
} }
void ServiceMessagePainter::paint(Painter &p, const HistoryService *message, const PaintContext &context, int height) { void ServiceMessagePainter::paint(Painter &p, const HistoryService *message, const PaintContext &context, int height) {
int left = 0, width = 0; auto g = message->countGeometry();
message->countPositionAndSize(left, width); if (g.width() < 1) return;
if (width < 1) return;
auto fullAnimMs = App::main() ? App::main()->animActiveTimeStart(message) : 0LL; auto fullAnimMs = App::main() ? App::main()->animActiveTimeStart(message) : 0LL;
if (fullAnimMs > 0 && fullAnimMs <= context.ms) { if (fullAnimMs > 0 && fullAnimMs <= context.ms) {
@ -215,20 +214,15 @@ void ServiceMessagePainter::paint(Painter &p, const HistoryService *message, con
if (auto media = message->getMedia()) { if (auto media = message->getMedia()) {
height -= st::msgServiceMargin.top() + media->height(); height -= st::msgServiceMargin.top() + media->height();
int32 left = st::msgServiceMargin.left() + (width - media->maxWidth()) / 2, top = st::msgServiceMargin.top() + height + st::msgServiceMargin.top(); auto left = st::msgServiceMargin.left() + (g.width() - media->maxWidth()) / 2, top = st::msgServiceMargin.top() + height + st::msgServiceMargin.top();
p.translate(left, top); p.translate(left, top);
media->draw(p, context.clip.translated(-left, -top), message->toMediaSelection(context.selection), context.ms); media->draw(p, context.clip.translated(-left, -top), message->toMediaSelection(context.selection), context.ms);
p.translate(-left, -top); p.translate(-left, -top);
} }
QRect trect(QRect(left, st::msgServiceMargin.top(), width, height).marginsAdded(-st::msgServicePadding)); auto trect = QRect(g.left(), st::msgServiceMargin.top(), g.width(), height).marginsAdded(-st::msgServicePadding);
paintComplexBubble(p, left, width, message->_text, trect); paintComplexBubble(p, g.left(), g.width(), message->_text, trect);
if (width > message->maxWidth()) {
left += (width - message->maxWidth()) / 2;
width = message->maxWidth();
}
p.setBrush(Qt::NoBrush); p.setBrush(Qt::NoBrush);
p.setPen(st::msgServiceFg); p.setPen(st::msgServiceFg);