Implement correct saved messages history layout.

Forwarded info is displayed as a message author info (name/photo).
Outgoing messages without forwarded info are displayed as out().
Messages with save_from_ info have a button for GoToOriginal().
This commit is contained in:
John Preston 2017-12-05 20:14:28 +04:00
parent f8e094392f
commit 85d8273009
13 changed files with 235 additions and 108 deletions

View File

@ -110,7 +110,7 @@ void History::clearLastKeyboard() {
} }
bool History::canHaveFromPhotos() const { bool History::canHaveFromPhotos() const {
if (peer->isUser() && !Adaptive::ChatWide()) { if (peer->isUser() && !peer->isSelf() && !Adaptive::ChatWide()) {
return false; return false;
} else if (isChannel() && !peer->isMegagroup()) { } else if (isChannel() && !peer->isMegagroup()) {
return false; return false;

View File

@ -458,5 +458,6 @@ historyFastShareSize: 31px;
historyFastShareLeft: 13px; historyFastShareLeft: 13px;
historyFastShareBottom: 5px; historyFastShareBottom: 5px;
historyFastShareIcon: icon {{ "fast_share", msgServiceFg, point(4px, 3px)}}; historyFastShareIcon: icon {{ "fast_share", msgServiceFg, point(4px, 3px)}};
historyGoToOriginalIcon: icon {{ "title_back-flip_horizontal", msgServiceFg, point(8px, 7px) }};
historySavedFont: font(semibold 15px); historySavedFont: font(semibold 15px);

View File

@ -527,7 +527,7 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
// paint the userpic if it intersects the painted rect // paint the userpic if it intersects the painted rect
if (userpicTop + st::msgPhotoSize > clip.top()) { if (userpicTop + st::msgPhotoSize > clip.top()) {
message->from()->paintUserpicLeft(p, st::historyPhotoLeft, userpicTop, message->history()->width, st::msgPhotoSize); message->displayFrom()->paintUserpicLeft(p, st::historyPhotoLeft, userpicTop, message->history()->width, st::msgPhotoSize);
} }
return true; return true;
}); });
@ -2140,7 +2140,7 @@ void HistoryInner::onUpdateSelected() {
// stop enumeration if we've found a userpic under the cursor // stop enumeration if we've found a userpic under the cursor
if (point.y() >= userpicTop && point.y() < userpicTop + st::msgPhotoSize) { if (point.y() >= userpicTop && point.y() < userpicTop + st::msgPhotoSize) {
dragState.link = message->from()->openLink(); dragState.link = message->displayFrom()->openLink();
lnkhost = message; lnkhost = message;
return false; return false;
} }

View File

@ -616,8 +616,7 @@ HistoryItem::HistoryItem(
, date(date) , date(date)
, _history(history) , _history(history)
, _from(from ? App::user(from) : history->peer) , _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) {
} }
void HistoryItem::finishCreate() { void HistoryItem::finishCreate() {
@ -789,11 +788,18 @@ void HistoryItem::nextItemChanged() {
bool HistoryItem::computeIsAttachToPrevious(not_null<HistoryItem*> previous) { bool HistoryItem::computeIsAttachToPrevious(not_null<HistoryItem*> previous) {
if (!Has<HistoryMessageDate>() && !Has<HistoryMessageUnreadBar>()) { if (!Has<HistoryMessageDate>() && !Has<HistoryMessageUnreadBar>()) {
return !isPost() && !previous->isPost() const auto possible = !isPost() && !previous->isPost()
&& !serviceMsg() && !previous->serviceMsg() && !serviceMsg() && !previous->serviceMsg()
&& !isEmpty() && !previous->isEmpty() && !isEmpty() && !previous->isEmpty()
&& previous->from() == from()
&& (qAbs(previous->date.secsTo(date)) < kAttachMessageToPreviousSecondsDelta); && (qAbs(previous->date.secsTo(date)) < kAttachMessageToPreviousSecondsDelta);
if (possible) {
if (history()->peer->isSelf()) {
return previous->senderOriginal() == senderOriginal()
&& (previous->Has<HistoryMessageForwarded>() == Has<HistoryMessageForwarded>());
} else {
return previous->from() == from();
}
}
} }
return false; return false;
} }
@ -1022,6 +1028,13 @@ QString HistoryItem::directLink() const {
return QString(); return QString();
} }
bool HistoryItem::hasOutLayout() const {
if (history()->peer->isSelf()) {
return !Has<HistoryMessageForwarded>();
}
return out() && !isPost();
}
bool HistoryItem::unread() const { bool HistoryItem::unread() const {
// Messages from myself are always read. // Messages from myself are always read.
if (history()->peer->isSelf()) return false; if (history()->peer->isSelf()) return false;

View File

@ -155,6 +155,9 @@ struct HistoryMessageForwarded : public RuntimeComponent<HistoryMessageForwarded
QString _originalAuthor; QString _originalAuthor;
MsgId _originalId = 0; MsgId _originalId = 0;
mutable Text _text = { 1 }; mutable Text _text = { 1 };
PeerData *_savedFromPeer = nullptr;
MsgId _savedFromMsgId = 0;
}; };
struct HistoryMessageReply : public RuntimeComponent<HistoryMessageReply> { struct HistoryMessageReply : public RuntimeComponent<HistoryMessageReply> {
@ -645,9 +648,7 @@ public:
bool isSilent() const { bool isSilent() const {
return _flags & MTPDmessage::Flag::f_silent; return _flags & MTPDmessage::Flag::f_silent;
} }
bool hasOutLayout() const { bool hasOutLayout() const;
return out() && !isPost();
}
virtual int32 viewsCount() const { virtual int32 viewsCount() const {
return hasViews() ? 1 : -1; return hasViews() ? 1 : -1;
} }
@ -727,13 +728,13 @@ public:
virtual void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const { virtual void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const {
} }
virtual ClickHandlerPtr fastShareLink() const { virtual ClickHandlerPtr rightActionLink() const {
return ClickHandlerPtr(); return ClickHandlerPtr();
} }
virtual bool displayFastShare() const { virtual bool displayRightAction() const {
return false; return false;
} }
virtual void drawFastShare(Painter &p, int left, int top, int outerWidth) const { virtual void drawRightAction(Painter &p, int left, int top, int outerWidth) const {
} }
virtual void setViewsCount(int32 count) { virtual void setViewsCount(int32 count) {
} }
@ -985,8 +986,6 @@ protected:
int _indexInBlock = -1; int _indexInBlock = -1;
MTPDmessage::Flags _flags = 0; MTPDmessage::Flags _flags = 0;
mutable int32 _authorNameVersion = 0;
HistoryItem *previousItem() const { HistoryItem *previousItem() const {
if (_block && _indexInBlock >= 0) { if (_block && _indexInBlock >= 0) {
if (_indexInBlock > 0) { if (_indexInBlock > 0) {

View File

@ -366,7 +366,6 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim
bool notChild = (_parent->getMedia() == this); bool notChild = (_parent->getMedia() == this);
int skipx = 0, skipy = 0, width = _width, height = _height; int skipx = 0, skipy = 0, width = _width, height = _height;
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost;
int captionw = width - st::msgPadding.left() - st::msgPadding.right(); int captionw = width - st::msgPadding.left() - st::msgPadding.right();
@ -462,6 +461,7 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim
// date // date
if (!_caption.isEmpty()) { if (!_caption.isEmpty()) {
auto outbg = _parent->hasOutLayout();
p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg));
_caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaPadding.bottom() + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); _caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaPadding.bottom() + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection);
} else if (notChild) { } else if (notChild) {
@ -470,10 +470,10 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim
if (_data->uploading() || App::hoveredItem() == _parent) { if (_data->uploading() || App::hoveredItem() == _parent) {
_parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage); _parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage);
} }
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
_parent->drawFastShare(p, fastShareLeft, fastShareTop, 2 * skipx + width); _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width);
} }
} }
} }
@ -523,11 +523,11 @@ HistoryTextState HistoryPhoto::getState(QPoint point, HistoryStateRequest reques
if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) {
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) { if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = _parent->fastShareLink(); result.link = _parent->rightActionLink();
} }
} }
} }
@ -779,7 +779,6 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, Tim
int skipx = 0, skipy = 0, width = _width, height = _height; int skipx = 0, skipy = 0, width = _width, height = _height;
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost;
int captionw = width - st::msgPadding.left() - st::msgPadding.right(); int captionw = width - st::msgPadding.left() - st::msgPadding.right();
@ -868,15 +867,16 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, Tim
// date // date
if (!_caption.isEmpty()) { if (!_caption.isEmpty()) {
auto outbg = _parent->hasOutLayout();
p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg));
_caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaPadding.bottom() + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection); _caption.draw(p, st::msgPadding.left(), skipy + height + st::mediaPadding.bottom() + st::mediaCaptionSkip, captionw, style::al_left, 0, -1, selection);
} else if (_parent->getMedia() == this) { } else if (_parent->getMedia() == this) {
auto fullRight = skipx + width, fullBottom = skipy + height; auto fullRight = skipx + width, fullBottom = skipy + height;
_parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage); _parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage);
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
_parent->drawFastShare(p, fastShareLeft, fastShareTop, 2 * skipx + width); _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width);
} }
} }
} }
@ -921,11 +921,11 @@ HistoryTextState HistoryVideo::getState(QPoint point, HistoryStateRequest reques
if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) {
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) { if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = _parent->fastShareLink(); result.link = _parent->rightActionLink();
} }
} }
} }
@ -1224,8 +1224,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection,
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
int captionw = _width - st::msgPadding.left() - st::msgPadding.right(); int captionw = _width - st::msgPadding.left() - st::msgPadding.right();
auto outbg = _parent->hasOutLayout();
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost;
if (displayLoading) { if (displayLoading) {
ensureAnimation(); ensureAnimation();
@ -1468,7 +1467,6 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result;
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost;
bool loaded = _data->loaded(); bool loaded = _data->loaded();
bool showPause = updateStatusText(); bool showPause = updateStatusText();
@ -2010,7 +2008,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
auto isChildMedia = (_parent->getMedia() != this); auto isChildMedia = (_parent->getMedia() != this);
auto captionw = width - st::msgPadding.left() - st::msgPadding.right(); auto captionw = width - st::msgPadding.left() - st::msgPadding.right();
@ -2055,8 +2053,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
auto forwarded = separateRoundVideo ? _parent->Get<HistoryMessageForwarded>() : nullptr; auto forwarded = separateRoundVideo ? _parent->Get<HistoryMessageForwarded>() : nullptr;
if (via || reply || forwarded) { if (via || reply || forwarded) {
usew = _maxw - additionalWidth(via, reply, forwarded); usew = _maxw - additionalWidth(via, reply, forwarded);
if (isPost) { if (outbg) {
} else if (out) {
usex = _width - usew; usex = _width - usew;
} }
} }
@ -2207,7 +2204,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
if (reply) { if (reply) {
recth += st::msgReplyBarSize.height(); recth += st::msgReplyBarSize.height();
} }
int rectx = isPost ? (usew + st::msgReplyPadding.left()) : (out ? 0 : (usew + st::msgReplyPadding.left())); int rectx = outbg ? 0 : (usew + st::msgReplyPadding.left());
int recty = skipy; int recty = skipy;
if (rtl()) rectx = _width - rectx - rectw; if (rtl()) rectx = _width - rectx - rectw;
@ -2260,14 +2257,14 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
if (isRound || _data->uploading() || App::hoveredItem() == _parent) { if (isRound || _data->uploading() || App::hoveredItem() == _parent) {
_parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, isRound ? InfoDisplayOverBackground : InfoDisplayOverImage); _parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, isRound ? InfoDisplayOverBackground : InfoDisplayOverImage);
} }
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (fastShareLeft + st::historyFastShareSize > maxRight) { if (fastShareLeft + st::historyFastShareSize > maxRight) {
fastShareLeft = (fullRight - st::historyFastShareSize - st::msgDateImgDelta); fastShareLeft = (fullRight - st::historyFastShareSize - st::msgDateImgDelta);
fastShareTop -= (st::msgDateImgDelta + st::msgDateImgPadding.y() + st::msgDateFont->height + st::msgDateImgPadding.y()); fastShareTop -= (st::msgDateImgDelta + st::msgDateImgPadding.y() + st::msgDateFont->height + st::msgDateImgPadding.y());
} }
_parent->drawFastShare(p, fastShareLeft, fastShareTop, 2 * skipx + width); _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width);
} }
} }
} }
@ -2297,7 +2294,7 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
width -= st::mediaPadding.left() + st::mediaPadding.right(); width -= st::mediaPadding.left() + st::mediaPadding.right();
height -= skipy + st::mediaPadding.bottom(); height -= skipy + st::mediaPadding.bottom();
} }
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
auto isChildMedia = (_parent->getMedia() != this); auto isChildMedia = (_parent->getMedia() != this);
auto isRound = _data->isRoundVideo(); auto isRound = _data->isRoundVideo();
auto usew = width, usex = 0; auto usew = width, usex = 0;
@ -2307,8 +2304,7 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
auto forwarded = separateRoundVideo ? _parent->Get<HistoryMessageForwarded>() : nullptr; auto forwarded = separateRoundVideo ? _parent->Get<HistoryMessageForwarded>() : nullptr;
if (via || reply || forwarded) { if (via || reply || forwarded) {
usew = _maxw - additionalWidth(via, reply, forwarded); usew = _maxw - additionalWidth(via, reply, forwarded);
if (isPost) { if (outbg) {
} else if (out) {
usex = _width - usew; usex = _width - usew;
} }
} }
@ -2328,7 +2324,7 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
if (reply) { if (reply) {
recth += st::msgReplyBarSize.height(); recth += st::msgReplyBarSize.height();
} }
auto rectx = isPost ? (usew + st::msgReplyPadding.left()) : (out ? 0 : (usew + st::msgReplyPadding.left())); auto rectx = outbg ? 0 : (usew + st::msgReplyPadding.left());
auto recty = skipy; auto recty = skipy;
if (rtl()) rectx = _width - rectx - rectw; if (rtl()) rectx = _width - rectx - rectw;
@ -2401,7 +2397,7 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
} }
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (fastShareLeft + st::historyFastShareSize > maxRight) { if (fastShareLeft + st::historyFastShareSize > maxRight) {
@ -2409,7 +2405,7 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
fastShareTop -= st::msgDateImgDelta + st::msgDateImgPadding.y() + st::msgDateFont->height + st::msgDateImgPadding.y(); fastShareTop -= st::msgDateImgDelta + st::msgDateImgPadding.y() + st::msgDateFont->height + st::msgDateImgPadding.y();
} }
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) { if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = _parent->fastShareLink(); result.link = _parent->rightActionLink();
} }
} }
} }
@ -2739,15 +2735,15 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T
bool loaded = _data->loaded(); bool loaded = _data->loaded();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
bool out = _parent->out(), isPost = _parent->isPost(), childmedia = (_parent->getMedia() != this); auto outbg = _parent->hasOutLayout();
auto childmedia = (_parent->getMedia() != this);
int usew = _maxw, usex = 0; int usew = _maxw, usex = 0;
auto via = childmedia ? nullptr : _parent->Get<HistoryMessageVia>(); auto via = childmedia ? nullptr : _parent->Get<HistoryMessageVia>();
auto reply = childmedia ? nullptr : _parent->Get<HistoryMessageReply>(); auto reply = childmedia ? nullptr : _parent->Get<HistoryMessageReply>();
if (via || reply) { if (via || reply) {
usew -= additionalWidth(via, reply); usew -= additionalWidth(via, reply);
if (isPost) { if (outbg) {
} else if (out) {
usex = _width - usew; usex = _width - usew;
} }
} }
@ -2780,7 +2776,7 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T
if (reply) { if (reply) {
recth += st::msgReplyBarSize.height(); recth += st::msgReplyBarSize.height();
} }
int rectx = isPost ? (usew + st::msgReplyPadding.left()) : (out ? 0 : (usew + st::msgReplyPadding.left())); int rectx = outbg ? 0 : (usew + st::msgReplyPadding.left());
int recty = st::msgDateImgDelta; int recty = st::msgDateImgDelta;
if (rtl()) rectx = _width - rectx - rectw; if (rtl()) rectx = _width - rectx - rectw;
@ -2802,10 +2798,10 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T
reply->paint(p, _parent, rectx, recty, rectw, flags); reply->paint(p, _parent, rectx, recty, rectw, flags);
} }
} }
if (_parent->displayFastShare()) { if (_parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
_parent->drawFastShare(p, fastShareLeft, fastShareTop, 2 * usex + usew); _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * usex + usew);
} }
} }
} }
@ -2814,15 +2810,15 @@ HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest requ
HistoryTextState result; HistoryTextState result;
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result;
bool out = _parent->out(), isPost = _parent->isPost(), childmedia = (_parent->getMedia() != this); auto outbg = _parent->hasOutLayout();
auto childmedia = (_parent->getMedia() != this);
int usew = _maxw, usex = 0; int usew = _maxw, usex = 0;
auto via = childmedia ? nullptr : _parent->Get<HistoryMessageVia>(); auto via = childmedia ? nullptr : _parent->Get<HistoryMessageVia>();
auto reply = childmedia ? nullptr : _parent->Get<HistoryMessageReply>(); auto reply = childmedia ? nullptr : _parent->Get<HistoryMessageReply>();
if (via || reply) { if (via || reply) {
usew -= additionalWidth(via, reply); usew -= additionalWidth(via, reply);
if (isPost) { if (outbg) {
} else if (out) {
usex = _width - usew; usex = _width - usew;
} }
} }
@ -2837,7 +2833,7 @@ HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest requ
if (reply) { if (reply) {
recth += st::msgReplyBarSize.height(); recth += st::msgReplyBarSize.height();
} }
int rectx = isPost ? (usew + st::msgReplyPadding.left()) : (out ? 0 : (usew + st::msgReplyPadding.left())); int rectx = outbg ? 0 : (usew + st::msgReplyPadding.left());
int recty = st::msgDateImgDelta; int recty = st::msgDateImgDelta;
if (rtl()) rectx = _width - rectx - rectw; if (rtl()) rectx = _width - rectx - rectw;
@ -2864,11 +2860,11 @@ HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest requ
if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) {
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
if (_parent->displayFastShare()) { if (_parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) { if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = _parent->fastShareLink(); result.link = _parent->rightActionLink();
} }
} }
} }
@ -3027,7 +3023,7 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return;
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
if (width >= _maxw) { if (width >= _maxw) {
@ -3082,7 +3078,6 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T
HistoryTextState HistoryContact::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryContact::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; HistoryTextState result;
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost;
int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0;
auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus; auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus;
@ -3193,7 +3188,7 @@ void HistoryCall::draw(Painter &p, const QRect &r, TextSelection selection, Time
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return;
auto skipx = 0, skipy = 0, width = _width, height = _height; auto skipx = 0, skipy = 0, width = _width, height = _height;
auto out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
auto selected = (selection == FullSelection); auto selected = (selection == FullSelection);
if (width >= _maxw) { if (width >= _maxw) {
@ -3549,7 +3544,7 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return;
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
auto &barfg = selected ? (outbg ? st::msgOutReplyBarSelColor : st::msgInReplyBarSelColor) : (outbg ? st::msgOutReplyBarColor : st::msgInReplyBarColor); auto &barfg = selected ? (outbg ? st::msgOutReplyBarSelColor : st::msgInReplyBarSelColor) : (outbg ? st::msgOutReplyBarColor : st::msgInReplyBarColor);
@ -3994,7 +3989,7 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return;
int32 width = _width, height = _height; int32 width = _width, height = _height;
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
auto &barfg = selected ? (outbg ? st::msgOutReplyBarSelColor : st::msgInReplyBarSelColor) : (outbg ? st::msgOutReplyBarColor : st::msgInReplyBarColor); auto &barfg = selected ? (outbg ? st::msgOutReplyBarSelColor : st::msgInReplyBarSelColor) : (outbg ? st::msgOutReplyBarColor : st::msgInReplyBarColor);
@ -4431,7 +4426,7 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return;
int32 width = _width, height = _height; int32 width = _width, height = _height;
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
auto &barfg = selected ? (outbg ? st::msgOutReplyBarSelColor : st::msgInReplyBarSelColor) : (outbg ? st::msgOutReplyBarColor : st::msgInReplyBarColor); auto &barfg = selected ? (outbg ? st::msgOutReplyBarSelColor : st::msgInReplyBarSelColor) : (outbg ? st::msgOutReplyBarColor : st::msgInReplyBarColor);
@ -4718,7 +4713,7 @@ void HistoryLocation::draw(Painter &p, const QRect &r, TextSelection selection,
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return;
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
bool out = _parent->out(), isPost = _parent->isPost(), outbg = out && !isPost; auto outbg = _parent->hasOutLayout();
bool selected = (selection == FullSelection); bool selected = (selection == FullSelection);
if (bubble) { if (bubble) {
@ -4781,10 +4776,10 @@ void HistoryLocation::draw(Painter &p, const QRect &r, TextSelection selection,
auto fullRight = skipx + width; auto fullRight = skipx + width;
auto fullBottom = _height - (skipx ? st::mediaPadding.bottom() : 0); auto fullBottom = _height - (skipx ? st::mediaPadding.bottom() : 0);
_parent->drawInfo(p, fullRight, fullBottom, skipx * 2 + width, selected, InfoDisplayOverImage); _parent->drawInfo(p, fullRight, fullBottom, skipx * 2 + width, selected, InfoDisplayOverImage);
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
_parent->drawFastShare(p, fastShareLeft, fastShareTop, 2 * skipx + width); _parent->drawRightAction(p, fastShareLeft, fastShareTop, 2 * skipx + width);
} }
} }
} }
@ -4843,11 +4838,11 @@ HistoryTextState HistoryLocation::getState(QPoint point, HistoryStateRequest req
if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) { if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) {
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
if (!bubble && _parent->displayFastShare()) { if (!bubble && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft); auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize); auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) { if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = _parent->fastShareLink(); result.link = _parent->rightActionLink();
} }
} }
} }

View File

@ -38,6 +38,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "styles/style_history.h" #include "styles/style_history.h"
#include "styles/style_window.h" #include "styles/style_window.h"
#include "window/notifications_manager.h" #include "window/notifications_manager.h"
#include "window/window_controller.h"
#include "observer_peer.h" #include "observer_peer.h"
#include "storage/storage_shared_media.h" #include "storage/storage_shared_media.h"
@ -636,9 +637,15 @@ HistoryMessage::HistoryMessage(not_null<History*> history, const MTPDmessage &ms
auto &f = msg.vfwd_from.c_messageFwdHeader(); auto &f = msg.vfwd_from.c_messageFwdHeader();
config.originalDate = ::date(f.vdate); config.originalDate = ::date(f.vdate);
if (f.has_from_id() || f.has_channel_id()) { if (f.has_from_id() || f.has_channel_id()) {
config.senderOriginal = f.has_channel_id() ? peerFromChannel(f.vchannel_id) : peerFromUser(f.vfrom_id); config.senderOriginal = f.has_channel_id()
? peerFromChannel(f.vchannel_id)
: peerFromUser(f.vfrom_id);
if (f.has_channel_post()) config.originalId = f.vchannel_post.v; if (f.has_channel_post()) config.originalId = f.vchannel_post.v;
if (f.has_post_author()) config.authorOriginal = qs(f.vpost_author); if (f.has_post_author()) config.authorOriginal = qs(f.vpost_author);
if (f.has_saved_from_peer() && f.has_saved_from_msg_id()) {
config.savedFromPeer = peerFromMTP(f.vsaved_from_peer);
config.savedFromMsgId = f.vsaved_from_msg_id.v;
}
} }
} }
if (msg.has_reply_to_msg_id()) config.replyTo = msg.vreply_to_msg_id.v; if (msg.has_reply_to_msg_id()) config.replyTo = msg.vreply_to_msg_id.v;
@ -676,7 +683,14 @@ HistoryMessage::HistoryMessage(not_null<History*> history, const MTPDmessageServ
setText(TextWithEntities {}); setText(TextWithEntities {});
} }
HistoryMessage::HistoryMessage(not_null<History*> history, MsgId id, MTPDmessage::Flags flags, QDateTime date, UserId from, const QString &postAuthor, not_null<HistoryMessage*> fwd) HistoryMessage::HistoryMessage(
not_null<History*> history,
MsgId id,
MTPDmessage::Flags flags,
QDateTime date,
UserId from,
const QString &postAuthor,
not_null<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;
@ -690,6 +704,19 @@ HistoryMessage::HistoryMessage(not_null<History*> history, MsgId id, MTPDmessage
config.originalId = fwd->idOriginal(); config.originalId = fwd->idOriginal();
} }
} }
if (history->peer->isSelf()) {
//
// iOS app sends you to the original post if we forward a forward from channel.
// But server returns not the original post but the forward in saved_from_...
//
//if (config.originalId) {
// config.savedFromPeer = config.senderOriginal;
// config.savedFromMsgId = config.originalId;
//} else {
config.savedFromPeer = fwd->history()->peer->id;
config.savedFromMsgId = fwd->id;
//}
}
if (flags & MTPDmessage::Flag::f_post_author) { if (flags & MTPDmessage::Flag::f_post_author) {
config.author = postAuthor; config.author = postAuthor;
} }
@ -728,14 +755,14 @@ HistoryMessage::HistoryMessage(not_null<History*> history, MsgId id, MTPDmessage
} }
HistoryMessage::HistoryMessage(not_null<History*> history, MsgId id, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, const TextWithEntities &textWithEntities) HistoryMessage::HistoryMessage(not_null<History*> history, MsgId id, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, 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, postAuthor, MTPnullMarkup); createComponentsHelper(flags, replyTo, viaBotId, postAuthor, MTPnullMarkup);
setText(textWithEntities); setText(textWithEntities);
} }
HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup) HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, 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, postAuthor, markup); createComponentsHelper(flags, replyTo, viaBotId, postAuthor, markup);
initMediaFromDocument(doc, caption); initMediaFromDocument(doc, caption);
@ -743,7 +770,7 @@ HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmess
} }
HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup) HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, 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, postAuthor, markup); createComponentsHelper(flags, replyTo, viaBotId, postAuthor, markup);
_media = std::make_unique<HistoryPhoto>(this, photo, caption); _media = std::make_unique<HistoryPhoto>(this, photo, caption);
@ -751,7 +778,7 @@ HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmess
} }
HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, GameData *game, const MTPReplyMarkup &markup) HistoryMessage::HistoryMessage(not_null<History*> history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, UserId viaBotId, QDateTime date, UserId from, const QString &postAuthor, 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, postAuthor, markup); createComponentsHelper(flags, replyTo, viaBotId, postAuthor, markup);
_media = std::make_unique<HistoryGame>(this, game); _media = std::make_unique<HistoryGame>(this, game);
@ -774,7 +801,10 @@ void HistoryMessage::updateMediaInBubbleState() {
auto mediaHasSomethingBelow = false; auto mediaHasSomethingBelow = false;
auto mediaHasSomethingAbove = false; auto mediaHasSomethingAbove = false;
auto getMediaHasSomethingAbove = [this] { auto getMediaHasSomethingAbove = [this] {
return displayFromName() || displayForwardedFrom() || Has<HistoryMessageReply>() || Has<HistoryMessageVia>(); return displayFromName()
|| displayForwardedFrom()
|| Has<HistoryMessageReply>()
|| Has<HistoryMessageVia>();
}; };
auto entry = Get<HistoryMessageLogEntryOriginal>(); auto entry = Get<HistoryMessageLogEntryOriginal>();
if (entry) { if (entry) {
@ -816,6 +846,12 @@ void HistoryMessage::updateMediaInBubbleState() {
_media->setInBubbleState(computeState()); _media->setInBubbleState(computeState());
} }
not_null<PeerData*> HistoryMessage::displayFrom() const {
return history()->peer->isSelf()
? senderOriginal()
: author();
}
void HistoryMessage::updateAdminBadgeState() { void HistoryMessage::updateAdminBadgeState() {
auto hasAdminBadge = [&] { auto hasAdminBadge = [&] {
if (auto channel = history()->peer->asChannel()) { if (auto channel = history()->peer->asChannel()) {
@ -863,6 +899,10 @@ bool HistoryMessage::uploading() const {
return _media && _media->uploading(); return _media && _media->uploading();
} }
bool HistoryMessage::displayRightAction() const {
return displayFastShare() || displayGoToOriginal();
}
bool HistoryMessage::displayFastShare() const { bool HistoryMessage::displayFastShare() const {
if (_history->peer->isChannel()) { if (_history->peer->isChannel()) {
return !_history->peer->isMegagroup(); return !_history->peer->isMegagroup();
@ -874,6 +914,15 @@ bool HistoryMessage::displayFastShare() const {
return false; return false;
} }
bool HistoryMessage::displayGoToOriginal() const {
if (_history->peer->isSelf()) {
if (auto forwarded = Get<HistoryMessageForwarded>()) {
return forwarded->_savedFromPeer && forwarded->_savedFromMsgId;
}
}
return false;
}
void HistoryMessage::createComponents(const CreateConfig &config) { void HistoryMessage::createComponents(const CreateConfig &config) {
uint64 mask = 0; uint64 mask = 0;
if (config.replyTo) { if (config.replyTo) {
@ -941,6 +990,8 @@ void HistoryMessage::createComponents(const CreateConfig &config) {
forwarded->_originalSender = App::peer(config.senderOriginal); forwarded->_originalSender = App::peer(config.senderOriginal);
forwarded->_originalId = config.originalId; forwarded->_originalId = config.originalId;
forwarded->_originalAuthor = config.authorOriginal; forwarded->_originalAuthor = config.authorOriginal;
forwarded->_savedFromPeer = App::peerLoaded(config.savedFromPeer);
forwarded->_savedFromMsgId = config.savedFromMsgId;
} }
if (auto markup = Get<HistoryMessageReplyMarkup>()) { if (auto markup = Get<HistoryMessageReplyMarkup>()) {
if (config.mtpMarkup) { if (config.mtpMarkup) {
@ -953,6 +1004,7 @@ void HistoryMessage::createComponents(const CreateConfig &config) {
} }
} }
initTime(); initTime();
_fromNameVersion = displayFrom()->nameVersion;
} }
QString formatViewsCount(int32 views) { QString formatViewsCount(int32 views) {
@ -1150,7 +1202,9 @@ void HistoryMessage::initDimensions() {
// Count parts in maxWidth(), don't count them in minHeight(). // Count parts in maxWidth(), don't count them in minHeight().
// They will be added in resizeGetHeight() anyway. // They will be added in resizeGetHeight() anyway.
if (displayFromName()) { if (displayFromName()) {
auto namew = st::msgPadding.left() + author()->nameText.maxWidth() + st::msgPadding.right(); auto namew = st::msgPadding.left()
+ displayFrom()->nameText.maxWidth()
+ st::msgPadding.right();
if (via && !forwarded) { if (via && !forwarded) {
namew += st::msgServiceFont->spacew + via->_maxWidth; namew += st::msgServiceFont->spacew + via->_maxWidth;
} }
@ -1210,13 +1264,21 @@ bool HistoryMessage::drawBubble() const {
return _media ? (!emptyText() || _media->needsBubble()) : !isEmpty(); return _media ? (!emptyText() || _media->needsBubble()) : !isEmpty();
} }
bool HistoryMessage::hasFromName() const {
return !hasOutLayout()
&& (!history()->peer->isUser() || history()->peer->isSelf());
}
QRect HistoryMessage::countGeometry() const { QRect HistoryMessage::countGeometry() const {
auto maxwidth = qMin(st::msgMaxWidth, _maxw); 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()));
} }
auto contentLeft = (!isPost() && out() && !Adaptive::ChatWide()) ? st::msgMargin.right() : st::msgMargin.left(); auto outLayout = hasOutLayout();
auto contentLeft = (outLayout && !Adaptive::ChatWide())
? st::msgMargin.right()
: st::msgMargin.left();
if (hasFromPhoto()) { if (hasFromPhoto()) {
contentLeft += 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) {
@ -1224,8 +1286,11 @@ QRect HistoryMessage::countGeometry() const {
} }
auto contentWidth = width() - st::msgMargin.left() - st::msgMargin.right(); auto contentWidth = width() - st::msgMargin.left() - st::msgMargin.right();
if (history()->peer->isSelf() && !outLayout) {
contentWidth -= st::msgPhotoSkip;
}
if (contentWidth > maxwidth) { if (contentWidth > maxwidth) {
if (!isPost() && out() && !Adaptive::ChatWide()) { if (outLayout && !Adaptive::ChatWide()) {
contentLeft += contentWidth - maxwidth; contentLeft += contentWidth - maxwidth;
} }
contentWidth = maxwidth; contentWidth = maxwidth;
@ -1241,7 +1306,7 @@ void HistoryMessage::fromNameUpdated(int32 width) const {
AdminBadgeText()); AdminBadgeText());
width -= st::msgPadding.right() + badgeWidth; width -= st::msgPadding.right() + badgeWidth;
} }
_authorNameVersion = author()->nameVersion; _fromNameVersion = displayFrom()->nameVersion;
if (!Has<HistoryMessageForwarded>()) { if (!Has<HistoryMessageForwarded>()) {
if (auto via = Get<HistoryMessageVia>()) { if (auto via = Get<HistoryMessageVia>()) {
via->resize(width via->resize(width
@ -1313,6 +1378,9 @@ void HistoryMessage::applyEditionToEmpty() {
bool HistoryMessage::displayForwardedFrom() const { bool HistoryMessage::displayForwardedFrom() const {
if (auto forwarded = Get<HistoryMessageForwarded>()) { if (auto forwarded = Get<HistoryMessageForwarded>()) {
if (history()->peer->isSelf()) {
return false;
}
return Has<HistoryMessageVia>() return Has<HistoryMessageVia>()
|| !_media || !_media
|| !_media->isDisplayed() || !_media->isDisplayed()
@ -1550,11 +1618,12 @@ int HistoryMessage::infoWidth() const {
} else if (id < 0 && history()->peer->isSelf()) { } else if (id < 0 && history()->peer->isSelf()) {
result += st::historySendStateSpace; result += st::historySendStateSpace;
} }
if (out() && !isPost()) { if (hasOutLayout()) {
result += st::historySendStateSpace; result += st::historySendStateSpace;
} }
return result; return result;
} }
int HistoryMessage::timeLeft() const { int HistoryMessage::timeLeft() const {
int result = 0; int result = 0;
if (auto views = Get<HistoryMessageViews>()) { if (auto views = Get<HistoryMessageViews>()) {
@ -1568,7 +1637,7 @@ int HistoryMessage::timeLeft() const {
void HistoryMessage::drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const { void HistoryMessage::drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const {
p.setFont(st::msgDateFont); p.setFont(st::msgDateFont);
bool outbg = out() && !isPost(); bool outbg = hasOutLayout();
bool invertedsprites = (type == InfoDisplayOverImage || type == InfoDisplayOverBackground); bool invertedsprites = (type == InfoDisplayOverImage || type == InfoDisplayOverBackground);
int32 infoRight = right, infoBottom = bottom; int32 infoRight = right, infoBottom = bottom;
switch (type) { switch (type) {
@ -1682,7 +1751,9 @@ void HistoryMessage::setId(MsgId newId) {
} }
void HistoryMessage::draw(Painter &p, QRect clip, 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); auto outbg = hasOutLayout();
auto bubble = drawBubble();
auto selected = (selection == FullSelection);
auto g = countGeometry(); auto g = countGeometry();
if (g.width() < 1) { if (g.width() < 1) {
@ -1733,7 +1804,7 @@ void HistoryMessage::draw(Painter &p, QRect clip, TextSelection selection, TimeM
} }
if (bubble) { if (bubble) {
if (displayFromName() && author()->nameVersion > _authorNameVersion) { if (displayFromName() && displayFrom()->nameVersion > _fromNameVersion) {
fromNameUpdated(g.width()); fromNameUpdated(g.width());
} }
@ -1799,10 +1870,10 @@ void HistoryMessage::draw(Painter &p, QRect clip, TextSelection selection, TimeM
if (needDrawInfo) { if (needDrawInfo) {
HistoryMessage::drawInfo(p, g.left() + g.width(), g.top() + g.height(), 2 * g.left() + g.width(), selected, InfoDisplayDefault); HistoryMessage::drawInfo(p, g.left() + g.width(), g.top() + g.height(), 2 * g.left() + g.width(), selected, InfoDisplayDefault);
} }
if (displayFastShare()) { if (displayRightAction()) {
auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft; auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft;
auto fastShareTop = g.top() + g.height() - st::historyFastShareBottom - st::historyFastShareSize; auto fastShareTop = g.top() + g.height() - st::historyFastShareBottom - st::historyFastShareSize;
drawFastShare(p, fastShareLeft, fastShareTop, width()); drawRightAction(p, fastShareLeft, fastShareTop, width());
} }
} else if (_media) { } else if (_media) {
p.translate(g.topLeft()); p.translate(g.topLeft());
@ -1818,7 +1889,7 @@ void HistoryMessage::draw(Painter &p, QRect clip, TextSelection selection, TimeM
} }
} }
void HistoryMessage::drawFastShare(Painter &p, int left, int top, int outerWidth) const { void HistoryMessage::drawRightAction(Painter &p, int left, int top, int outerWidth) const {
{ {
p.setPen(Qt::NoPen); p.setPen(Qt::NoPen);
p.setBrush(st::msgServiceBg); p.setBrush(st::msgServiceBg);
@ -1826,7 +1897,11 @@ void HistoryMessage::drawFastShare(Painter &p, int left, int top, int outerWidth
PainterHighQualityEnabler hq(p); PainterHighQualityEnabler hq(p);
p.drawEllipse(rtlrect(left, top, st::historyFastShareSize, st::historyFastShareSize, outerWidth)); p.drawEllipse(rtlrect(left, top, st::historyFastShareSize, st::historyFastShareSize, outerWidth));
} }
st::historyFastShareIcon.paint(p, left, top, outerWidth); if (displayFastShare()) {
st::historyFastShareIcon.paint(p, left, top, outerWidth);
} else {
st::historyGoToOriginalIcon.paint(p, left, top, outerWidth);
}
} }
void HistoryMessage::paintFromName(Painter &p, QRect &trect, bool selected) const { void HistoryMessage::paintFromName(Painter &p, QRect &trect, bool selected) const {
@ -1849,7 +1924,7 @@ void HistoryMessage::paintFromName(Painter &p, QRect &trect, bool selected) cons
} else { } else {
p.setPen(FromNameFg(author(), selected)); p.setPen(FromNameFg(author(), selected));
} }
author()->nameText.drawElided(p, availableLeft, trect.top(), availableWidth); displayFrom()->nameText.drawElided(p, availableLeft, trect.top(), availableWidth);
auto skipWidth = author()->nameText.maxWidth() + st::msgServiceFont->spacew; auto skipWidth = author()->nameText.maxWidth() + st::msgServiceFont->spacew;
availableLeft += skipWidth; availableLeft += skipWidth;
availableWidth -= skipWidth; availableWidth -= skipWidth;
@ -1857,7 +1932,7 @@ void HistoryMessage::paintFromName(Painter &p, QRect &trect, bool selected) cons
auto forwarded = Get<HistoryMessageForwarded>(); auto forwarded = Get<HistoryMessageForwarded>();
auto via = Get<HistoryMessageVia>(); auto via = Get<HistoryMessageVia>();
if (via && !forwarded && availableWidth > 0) { if (via && !forwarded && availableWidth > 0) {
auto outbg = out() && !isPost(); auto outbg = hasOutLayout();
p.setPen(selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg)); p.setPen(selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg));
p.drawText(availableLeft, trect.top() + st::msgServiceFont->ascent, via->_text); p.drawText(availableLeft, trect.top() + st::msgServiceFont->ascent, via->_text);
auto skipWidth = via->_width + st::msgServiceFont->spacew; auto skipWidth = via->_width + st::msgServiceFont->spacew;
@ -1920,7 +1995,7 @@ void HistoryMessage::paintViaBotIdInfo(Painter &p, QRect &trect, bool selected)
} }
void HistoryMessage::paintText(Painter &p, QRect &trect, TextSelection selection) const { void HistoryMessage::paintText(Painter &p, QRect &trect, TextSelection selection) const {
auto outbg = out() && !isPost(); auto outbg = hasOutLayout();
auto selected = (selection == FullSelection); auto selected = (selection == FullSelection);
p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg)); p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg));
p.setFont(st::msgFont); p.setFont(st::msgFont);
@ -1958,6 +2033,9 @@ int HistoryMessage::performResizeGetHeight() {
if (width() < st::msgMinWidth) return _height; if (width() < st::msgMinWidth) return _height;
auto contentWidth = width() - (st::msgMargin.left() + st::msgMargin.right()); auto contentWidth = width() - (st::msgMargin.left() + st::msgMargin.right());
if (history()->peer->isSelf() && !hasOutLayout()) {
contentWidth -= st::msgPhotoSkip;
}
if (contentWidth < st::msgPadding.left() + st::msgPadding.right() + 1) { if (contentWidth < st::msgPadding.left() + st::msgPadding.right() + 1) {
contentWidth = st::msgPadding.left() + st::msgPadding.right() + 1; contentWidth = st::msgPadding.left() + st::msgPadding.right() + 1;
} else if (contentWidth > st::msgMaxWidth) { } else if (contentWidth > st::msgMaxWidth) {
@ -2157,11 +2235,11 @@ HistoryTextState HistoryMessage::getState(QPoint point, HistoryStateRequest requ
result.cursor = HistoryInDateCursorState; result.cursor = HistoryInDateCursorState;
} }
} }
if (displayFastShare()) { if (displayRightAction()) {
auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft; auto fastShareLeft = g.left() + g.width() + st::historyFastShareLeft;
auto fastShareTop = g.top() + g.height() - st::historyFastShareBottom - st::historyFastShareSize; auto fastShareTop = g.top() + g.height() - st::historyFastShareBottom - st::historyFastShareSize;
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) { if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = fastShareLink(); result.link = rightActionLink();
} }
} }
} else if (_media) { } else if (_media) {
@ -2180,15 +2258,26 @@ HistoryTextState HistoryMessage::getState(QPoint point, HistoryStateRequest requ
return result; return result;
} }
ClickHandlerPtr HistoryMessage::fastShareLink() const { ClickHandlerPtr HistoryMessage::rightActionLink() const {
if (!_fastShareLink) { if (!_rightActionLink) {
_fastShareLink = MakeShared<LambdaClickHandler>([id = fullId()] { const auto itemId = fullId();
if (auto item = App::histItemById(id)) { const auto forwarded = Get<HistoryMessageForwarded>();
FastShareMessage(item->toHistoryMessage()); const auto savedFromPeer = forwarded ? forwarded->_savedFromPeer : nullptr;
const auto savedFromMsgId = forwarded ? forwarded->_savedFromMsgId : 0;
_rightActionLink = MakeShared<LambdaClickHandler>([=] {
if (auto item = App::histItemById(itemId)) {
if (savedFromPeer && savedFromMsgId) {
App::wnd()->controller()->showPeerHistory(
savedFromPeer,
Window::SectionShow::Way::Forward,
savedFromMsgId);
} else {
FastShareMessage(item->toHistoryMessage());
}
} }
}); });
} }
return _fastShareLink; return _rightActionLink;
} }
// Forward to _media. // Forward to _media.
@ -2245,13 +2334,14 @@ void HistoryMessage::updatePressed(QPoint point) {
bool HistoryMessage::getStateFromName(QPoint point, QRect &trect, HistoryTextState *outResult) const { bool HistoryMessage::getStateFromName(QPoint point, QRect &trect, HistoryTextState *outResult) const {
if (displayFromName()) { if (displayFromName()) {
if (point.y() >= trect.top() && point.y() < trect.top() + st::msgNameFont->height) { if (point.y() >= trect.top() && point.y() < trect.top() + st::msgNameFont->height) {
if (point.x() >= trect.left() && point.x() < trect.left() + trect.width() && point.x() < trect.left() + author()->nameText.maxWidth()) { auto user = displayFrom();
outResult->link = author()->openLink(); if (point.x() >= trect.left() && point.x() < trect.left() + trect.width() && point.x() < trect.left() + user->nameText.maxWidth()) {
outResult->link = user->openLink();
return true; return true;
} }
auto forwarded = Get<HistoryMessageForwarded>(); auto forwarded = Get<HistoryMessageForwarded>();
auto via = Get<HistoryMessageVia>(); auto via = Get<HistoryMessageVia>();
if (via && !forwarded && point.x() >= trect.left() + author()->nameText.maxWidth() + st::msgServiceFont->spacew && point.x() < trect.left() + author()->nameText.maxWidth() + st::msgServiceFont->spacew + via->_width) { if (via && !forwarded && point.x() >= trect.left() + author()->nameText.maxWidth() + st::msgServiceFont->spacew && point.x() < trect.left() + user->nameText.maxWidth() + st::msgServiceFont->spacew + via->_width) {
outResult->link = via->_lnk; outResult->link = via->_lnk;
return true; return true;
} }
@ -2375,7 +2465,14 @@ bool HistoryMessage::displayFromPhoto() const {
} }
bool HistoryMessage::hasFromPhoto() const { bool HistoryMessage::hasFromPhoto() const {
return (Adaptive::ChatWide() || (!out() && !history()->peer->isUser())) && !isPost() && !isEmpty(); if (isPost() || isEmpty()) {
return false;
} else if (Adaptive::ChatWide()) {
return true;
} else if (history()->peer->isSelf()) {
return Has<HistoryMessageForwarded>();
}
return !out() && !history()->peer->isUser();
} }
HistoryMessage::~HistoryMessage() { HistoryMessage::~HistoryMessage() {

View File

@ -62,9 +62,7 @@ public:
bool hasBubble() const override { bool hasBubble() const override {
return drawBubble(); return drawBubble();
} }
bool hasFromName() const { bool hasFromName() const;
return (!out() || isPost()) && !history()->peer->isUser();
}
bool displayFromName() const { bool displayFromName() const {
if (!hasFromName()) return false; if (!hasFromName()) return false;
if (isAttachedToPrevious()) return false; if (isAttachedToPrevious()) return false;
@ -72,17 +70,17 @@ public:
} }
bool displayEditedBadge(bool hasViaBotOrInlineMarkup) const; bool displayEditedBadge(bool hasViaBotOrInlineMarkup) const;
bool uploading() const; bool uploading() const;
bool displayFastShare() const override; bool displayRightAction() const override;
void applyGroupAdminChanges( void applyGroupAdminChanges(
const base::flat_map<UserId, bool> &changes) override; const base::flat_map<UserId, bool> &changes) override;
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 drawFastShare(Painter &p, int left, int top, int outerWidth) const override; void drawRightAction(Painter &p, int left, int top, int outerWidth) 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, QRect clip, TextSelection selection, TimeMs ms) const override; void draw(Painter &p, QRect clip, TextSelection selection, TimeMs ms) const override;
ClickHandlerPtr fastShareLink() const override; ClickHandlerPtr rightActionLink() const override;
void dependencyItemRemoved(HistoryItem *dependency) override; void dependencyItemRemoved(HistoryItem *dependency) override;
@ -129,6 +127,8 @@ public:
return HistoryItem::viewsCount(); return HistoryItem::viewsCount();
} }
not_null<PeerData*> displayFrom() const;
bool updateDependencyItem() override { bool updateDependencyItem() override {
if (auto reply = Get<HistoryMessageReply>()) { if (auto reply = Get<HistoryMessageReply>()) {
return reply->updateData(this, true); return reply->updateData(this, true);
@ -191,10 +191,14 @@ private:
void setMedia(const MTPMessageMedia *media); void setMedia(const MTPMessageMedia *media);
void setReplyMarkup(const MTPReplyMarkup *markup); void setReplyMarkup(const MTPReplyMarkup *markup);
bool displayFastShare() const;
bool displayGoToOriginal() const;
QString _timeText; QString _timeText;
int _timeWidth = 0; int _timeWidth = 0;
mutable ClickHandlerPtr _fastShareLink; mutable ClickHandlerPtr _rightActionLink;
mutable int32 _fromNameVersion = 0;
struct CreateConfig { struct CreateConfig {
MsgId replyTo = 0; MsgId replyTo = 0;
@ -203,6 +207,8 @@ private:
QString author; QString author;
PeerId senderOriginal = 0; PeerId senderOriginal = 0;
MsgId originalId = 0; MsgId originalId = 0;
PeerId savedFromPeer = 0;
MsgId savedFromMsgId = 0;
QString authorOriginal; QString authorOriginal;
QDateTime originalDate; QDateTime originalDate;
QDateTime editDate; QDateTime editDate;

View File

@ -3882,7 +3882,7 @@ void HistoryWidget::forwardMessage() {
auto items = SelectedItemSet(); auto items = SelectedItemSet();
items.insert(item->id, item); items.insert(item->id, item);
App::main()->showForwardLayer(items); App::main()->showForwardBox(std::move(items));
} }
void HistoryWidget::selectMessage() { void HistoryWidget::selectMessage() {
@ -5280,7 +5280,7 @@ void HistoryWidget::mousePressEvent(QMouseEvent *e) {
if (readyToForward()) { if (readyToForward()) {
auto items = _toForward; auto items = _toForward;
App::main()->cancelForwarding(_history); App::main()->cancelForwarding(_history);
App::main()->showForwardLayer(items); App::main()->showForwardBox(std::move(items));
} else { } else {
Ui::showPeerHistory(_peer, _editMsgId ? _editMsgId : replyToId()); Ui::showPeerHistory(_peer, _editMsgId ? _editMsgId : replyToId());
} }
@ -6178,7 +6178,7 @@ void HistoryWidget::handlePeerUpdate() {
void HistoryWidget::onForwardSelected() { void HistoryWidget::onForwardSelected() {
if (!_list) return; if (!_list) return;
App::main()->showForwardLayer(getSelectedItems()); App::main()->showForwardBox(getSelectedItems());
} }
void HistoryWidget::confirmDeleteContextItem() { void HistoryWidget::confirmDeleteContextItem() {

View File

@ -485,6 +485,7 @@ private:
void showNextUnreadMention(); void showNextUnreadMention();
void handlePeerUpdate(); void handlePeerUpdate();
void setMembersShowAreaActive(bool active); void setMembersShowAreaActive(bool active);
void forwardItems(SelectedItemSet &&items);
void highlightMessage(MsgId universalMessageId); void highlightMessage(MsgId universalMessageId);
void adjustHighlightedMessageToMigrated(); void adjustHighlightedMessageToMigrated();

View File

@ -1013,6 +1013,20 @@ void MainWidget::showSendPathsLayer() {
hiderLayer(object_ptr<HistoryHider>(this)); hiderLayer(object_ptr<HistoryHider>(this));
} }
void MainWidget::showForwardBox(SelectedItemSet &&items) {
auto controller = std::make_unique<ChooseRecipientBoxController>(
[items = std::move(items)](not_null<PeerData*> peer) {
App::main()->setForwardDraft(peer->id, items);
});
Ui::show(Box<PeerListBox>(
std::move(controller),
[](not_null<PeerListBox*> box) {
box->addButton(langFactory(lng_cancel), [box] {
box->closeBox();
});
}));
}
void MainWidget::deleteLayer(int selectedCount) { void MainWidget::deleteLayer(int selectedCount) {
if (selectedCount) { if (selectedCount) {
auto forDelete = true; auto forDelete = true;

View File

@ -180,6 +180,7 @@ public:
void showForwardLayer(const SelectedItemSet &items); void showForwardLayer(const SelectedItemSet &items);
void showSendPathsLayer(); void showSendPathsLayer();
void showForwardBox(SelectedItemSet &&items);
void deleteLayer(int selectedCount = 0); // 0 - context item void deleteLayer(int selectedCount = 0); // 0 - context item
void cancelUploadLayer(); void cancelUploadLayer();
void shareContactLayer(UserData *contact); void shareContactLayer(UserData *contact);

View File

@ -939,7 +939,7 @@ void MediaView::onForward() {
if (auto main = App::main()) { if (auto main = App::main()) {
auto items = SelectedItemSet(); auto items = SelectedItemSet();
items.insert(item->id, item); items.insert(item->id, item);
main->showForwardLayer(items); main->showForwardBox(std::move(items));
} }
} }