Track multiple pinned messages in MessagesList.

This commit is contained in:
John Preston 2020-10-09 15:56:33 +03:00
parent 399b03beb2
commit ec35e3f081
19 changed files with 435 additions and 107 deletions

View file

@ -435,6 +435,8 @@ PRIVATE
data/data_photo.h
data/data_photo_media.cpp
data/data_photo_media.h
data/data_pinned_messages.cpp
data/data_pinned_messages.h
data/data_poll.cpp
data/data_poll.h
data/data_pts_waiter.cpp

View file

@ -1083,6 +1083,17 @@ void Updates::applyUpdateNoPtsCheck(const MTPUpdate &update) {
_session->data().updateEditedMessage(d.vmessage());
} break;
case mtpc_updatePinnedChannelMessages: {
const auto &d = update.c_updatePinnedChannelMessages();
const auto channelId = d.vchannel_id().v;
for (const auto &msgId : d.vmessages().v) {
const auto item = session().data().message(channelId, msgId.v);
if (item) {
item->setIsPinned(d.is_pinned());
}
}
} break;
case mtpc_updateEditMessage: {
auto &d = update.c_updateEditMessage();
_session->data().updateEditedMessage(d.vmessage());
@ -1098,6 +1109,17 @@ void Updates::applyUpdateNoPtsCheck(const MTPUpdate &update) {
_session->data().processMessagesDeleted(d.vchannel_id().v, d.vmessages().v);
} break;
case mtpc_updatePinnedMessages: {
const auto &d = update.c_updatePinnedMessages();
const auto peerId = peerFromMTP(d.vpeer());
for (const auto &msgId : d.vmessages().v) {
const auto item = session().data().message(0, msgId.v);
if (item) {
item->setIsPinned(d.is_pinned());
}
}
} break;
default: Unexpected("Type in applyUpdateNoPtsCheck()");
}
}
@ -1393,6 +1415,21 @@ void Updates::feedUpdate(const MTPUpdate &update) {
}
} break;
case mtpc_updatePinnedChannelMessages: {
auto &d = update.c_updatePinnedChannelMessages();
auto channel = session().data().channelLoaded(d.vchannel_id().v);
if (channel && !_handlingChannelDifference) {
if (channel->ptsRequesting()) { // skip global updates while getting channel difference
return;
} else {
channel->ptsUpdateAndApply(d.vpts().v, d.vpts_count().v, update);
}
} else {
applyUpdateNoPtsCheck(update);
}
} break;
// Messages being read.
case mtpc_updateReadHistoryInbox: {
auto &d = update.c_updateReadHistoryInbox();
@ -1998,6 +2035,12 @@ void Updates::feedUpdate(const MTPUpdate &update) {
}
} break;
// Pinned message.
case mtpc_updatePinnedMessages: {
const auto &d = update.c_updatePinnedMessages();
updateAndApply(d.vpts().v, d.vpts_count().v, update);
} break;
////// Cloud sticker sets
case mtpc_updateNewStickerSet: {
const auto &d = update.c_updateNewStickerSet();

View file

@ -444,14 +444,20 @@ PinMessageBox::PinMessageBox(
: _peer(peer)
, _api(&peer->session().mtp())
, _msgId(msgId)
, _text(this, tr::lng_pinned_pin_sure(tr::now), st::boxLabel) {
, _pinningOld(msgId < peer->topPinnedMessageId())
, _text(
this,
(_pinningOld
? "Do you want to pin an older message while leaving a more recent one pinned?" // #TODO pinned
: tr::lng_pinned_pin_sure(tr::now)),
st::boxLabel) {
}
void PinMessageBox::prepare() {
addButton(tr::lng_pinned_pin(), [this] { pinMessage(); });
addButton(tr::lng_cancel(), [this] { closeBox(); });
if (_peer->isChat() || _peer->isMegagroup()) {
if (!_pinningOld && (_peer->isChat() || _peer->isMegagroup())) {
_notify.create(this, tr::lng_pinned_notify(tr::now), true, st::defaultBoxCheckbox);
}

View file

@ -183,7 +183,8 @@ private:
const not_null<PeerData*> _peer;
MTP::Sender _api;
MsgId _msgId;
MsgId _msgId = 0;
bool _pinningOld = false;
object_ptr<Ui::FlatLabel> _text;
object_ptr<Ui::Checkbox> _notify = { nullptr };

View file

@ -383,9 +383,7 @@ void ChannelData::setUnavailableReasons(
void ChannelData::setAvailableMinId(MsgId availableMinId) {
if (_availableMinId != availableMinId) {
_availableMinId = availableMinId;
if (pinnedMessageId() <= _availableMinId) {
clearPinnedMessage();
}
clearPinnedMessages(_availableMinId + 1);
}
}
@ -772,9 +770,9 @@ void ApplyChannelUpdate(
}
}
if (const auto pinned = update.vpinned_msg_id()) {
channel->setPinnedMessageId(pinned->v);
channel->setTopPinnedMessageId(pinned->v);
} else {
channel->clearPinnedMessage();
channel->clearPinnedMessages();
}
if (channel->isMegagroup()) {
const auto stickerSet = update.vstickerset();

View file

@ -333,9 +333,9 @@ void ApplyChatUpdate(not_null<ChatData*> chat, const MTPDchatFull &update) {
return QString();
}));
if (const auto pinned = update.vpinned_msg_id()) {
chat->setPinnedMessageId(pinned->v);
chat->setTopPinnedMessageId(pinned->v);
} else {
chat->clearPinnedMessage();
chat->clearPinnedMessages();
}
chat->checkFolder(update.vfolder_id().value_or_empty());
chat->fullUpdated();

View file

@ -117,6 +117,11 @@ void MessagesList::addRange(
_sliceUpdated.fire(std::move(update));
}
void MessagesList::addOne(MessagePosition messageId) {
auto range = { messageId };
addRange(range, { messageId, messageId }, std::nullopt, true);
}
void MessagesList::addNew(MessagePosition messageId) {
auto range = { messageId };
addRange(range, { messageId, MaxMessagePosition }, std::nullopt, true);
@ -165,6 +170,31 @@ void MessagesList::removeAll(ChannelId channelId) {
}
}
void MessagesList::removeLessThan(MessagePosition messageId) {
auto removed = 0;
for (auto i = begin(_slices); i != end(_slices);) {
if (i->range.till <= messageId) {
removed += i->messages.size();
i = _slices.erase(i);
continue;
} else if (i->range.from <= messageId) {
_slices.modify(i, [&](Slice &slice) {
slice.range.from = MinMessagePosition;
auto from = begin(slice.messages);
auto till = ranges::lower_bound(slice.messages, messageId);
if (from != till) {
removed += till - from;
slice.messages.erase(from, till);
}
});
break;
}
}
if (removed && _count) {
*_count -= removed;
}
}
void MessagesList::invalidate() {
_slices.clear();
_count = std::nullopt;
@ -184,19 +214,26 @@ void MessagesList::invalidateBottom() {
_count = std::nullopt;
}
MessagesResult MessagesList::queryCurrent(const MessagesQuery &query) const {
if (!query.aroundId) {
return MessagesResult();
}
const auto slice = ranges::lower_bound(
_slices,
query.aroundId,
std::less<>(),
[](const Slice &slice) { return slice.range.till; });
return (slice != _slices.end() && slice->range.from <= query.aroundId)
? queryFromSlice(query, *slice)
: MessagesResult();
}
rpl::producer<MessagesResult> MessagesList::query(
MessagesQuery &&query) const {
return [this, query = std::move(query)](auto consumer) {
auto slice = query.aroundId
? ranges::lower_bound(
_slices,
query.aroundId,
std::less<>(),
[](const Slice &slice) { return slice.range.till; })
: _slices.end();
if (slice != _slices.end()
&& slice->range.from <= query.aroundId) {
consumer.put_next(queryFromSlice(query, *slice));
auto current = queryCurrent(query);
if (current.count.has_value() || !current.messageIds.empty()) {
consumer.put_next(std::move(current));
}
consumer.put_done();
return rpl::lifetime();
@ -207,6 +244,31 @@ rpl::producer<MessagesSliceUpdate> MessagesList::sliceUpdated() const {
return _sliceUpdated.events();
}
MessagesResult MessagesList::snapshot(MessagesQuery &&query) const {
return queryCurrent(query);
}
bool MessagesList::empty() const {
for (const auto &slice : _slices) {
if (!slice.messages.empty()) {
return false;
}
}
return true;
}
rpl::producer<MessagesResult> MessagesList::viewer(
MessagesQuery &&query) const {
auto copy = query;
return rpl::single(
queryCurrent(query)
) | rpl::then(sliceUpdated() | rpl::map([=] {
return queryCurrent(query);
})) | rpl::filter([=](const MessagesResult &value) {
return value.count.has_value() || !value.messageIds.empty();
});
}
MessagesResult MessagesList::queryFromSlice(
const MessagesQuery &query,
const Slice &slice) const {
@ -372,7 +434,7 @@ void MessagesSliceBuilder::mergeSliceData(
if (count) {
_fullCount = count;
}
const auto impossible = MessagePosition(-1, FullMsgId());
const auto impossible = MessagePosition{ .fullId = {}, .date = -1 };
auto wasMinId = _ids.empty() ? impossible : _ids.front();
auto wasMaxId = _ids.empty() ? impossible : _ids.back();
_ids.merge(messageIds.begin(), messageIds.end());

View file

@ -16,11 +16,8 @@ enum class LoadDirection : char {
};
struct MessagePosition {
constexpr MessagePosition() = default;
constexpr MessagePosition(TimeId date, FullMsgId fullId)
: fullId(fullId)
, date(date) {
}
FullMsgId fullId;
TimeId date = 0;
explicit operator bool() const {
return (fullId.msg != 0);
@ -50,18 +47,11 @@ struct MessagePosition {
inline constexpr bool operator!=(const MessagePosition &other) const {
return !(*this == other);
}
FullMsgId fullId;
TimeId date = 0;
};
struct MessagesRange {
constexpr MessagesRange() = default;
constexpr MessagesRange(MessagePosition from, MessagePosition till)
: from(from)
, till(till) {
}
MessagePosition from;
MessagePosition till;
inline constexpr bool operator==(const MessagesRange &other) const {
return (from == other.from)
@ -70,26 +60,26 @@ struct MessagesRange {
inline constexpr bool operator!=(const MessagesRange &other) const {
return !(*this == other);
}
MessagePosition from;
MessagePosition till;
};
constexpr auto MinDate = TimeId(0);
constexpr auto MaxDate = std::numeric_limits<TimeId>::max();
constexpr auto MinMessagePosition = MessagePosition(
MinDate,
FullMsgId(NoChannel, 1));
constexpr auto MaxMessagePosition = MessagePosition(
MaxDate,
FullMsgId(NoChannel, ServerMaxMsgId - 1));
constexpr auto FullMessagesRange = MessagesRange(
MinMessagePosition,
MaxMessagePosition);
constexpr auto UnreadMessagePosition = MessagePosition(
MinDate,
FullMsgId(NoChannel, ShowAtUnreadMsgId));
constexpr auto MinMessagePosition = MessagePosition{
.fullId = FullMsgId(NoChannel, 1),
.date = MinDate,
};
constexpr auto MaxMessagePosition = MessagePosition{
.fullId = FullMsgId(NoChannel, ServerMaxMsgId - 1),
.date = MaxDate,
};
constexpr auto FullMessagesRange = MessagesRange{
.from = MinMessagePosition,
.till = MaxMessagePosition,
};
constexpr auto UnreadMessagePosition = MessagePosition{
.fullId = FullMsgId(NoChannel, ShowAtUnreadMsgId),
.date = MinDate,
};
struct MessagesSlice {
std::vector<FullMsgId> ids;
@ -100,15 +90,6 @@ struct MessagesSlice {
};
struct MessagesQuery {
MessagesQuery(
MessagePosition aroundId,
int limitBefore,
int limitAfter)
: aroundId(aroundId)
, limitBefore(limitBefore)
, limitAfter(limitAfter) {
}
MessagePosition aroundId;
int limitBefore = 0;
int limitAfter = 0;
@ -129,6 +110,7 @@ struct MessagesSliceUpdate {
class MessagesList {
public:
void addOne(MessagePosition messageId);
void addNew(MessagePosition messageId);
void addSlice(
std::vector<MessagePosition> &&messageIds,
@ -136,10 +118,18 @@ public:
std::optional<int> count);
void removeOne(MessagePosition messageId);
void removeAll(ChannelId channelId);
void removeLessThan(MessagePosition messageId);
void invalidate();
void invalidateBottom();
rpl::producer<MessagesResult> query(MessagesQuery &&query) const;
rpl::producer<MessagesSliceUpdate> sliceUpdated() const;
[[nodiscard]] rpl::producer<MessagesResult> query(
MessagesQuery &&query) const;
[[nodiscard]] rpl::producer<MessagesSliceUpdate> sliceUpdated() const;
[[nodiscard]] MessagesResult snapshot(MessagesQuery &&query) const;
[[nodiscard]] rpl::producer<MessagesResult> viewer(
MessagesQuery &&query) const;
[[nodiscard]] bool empty() const;
private:
struct Slice {
@ -183,6 +173,7 @@ private:
MessagesResult queryFromSlice(
const MessagesQuery &query,
const Slice &slice) const;
MessagesResult queryCurrent(const MessagesQuery &query) const;
std::optional<int> _count;
base::flat_set<Slice> _slices;

View file

@ -16,6 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_session.h"
#include "data/data_file_origin.h"
#include "data/data_histories.h"
#include "data/data_pinned_messages.h"
#include "base/unixtime.h"
#include "base/crc32hash.h"
#include "lang/lang_keys.h"
@ -460,18 +461,83 @@ bool PeerData::canEditMessagesIndefinitely() const {
Unexpected("Peer type in PeerData::canEditMessagesIndefinitely.");
}
void PeerData::setPinnedMessageId(MsgId messageId) {
MsgId PeerData::topPinnedMessageId() const {
return _pinnedMessages ? _pinnedMessages->topId() : 0;
}
void PeerData::ensurePinnedMessagesCreated() {
if (!_pinnedMessages) {
_pinnedMessages = std::make_unique<Data::PinnedMessages>(
peerToChannel(id));
}
}
void PeerData::removeEmptyPinnedMessages() {
if (_pinnedMessages && _pinnedMessages->empty()) {
_pinnedMessages = nullptr;
}
}
bool PeerData::messageIdTooSmall(MsgId messageId) const {
if (const auto channel = asChannel()) {
return (messageId <= channel->availableMinId());
}
return false;
}
void PeerData::setTopPinnedMessageId(MsgId messageId) {
if (messageIdTooSmall(messageId)) {
return;
}
ensurePinnedMessagesCreated();
_pinnedMessages->setTopId(messageId);
}
void PeerData::clearPinnedMessages(MsgId lessThanId) {
if (!_pinnedMessages) {
return;
}
_pinnedMessages->clearLessThanId(lessThanId);
removeEmptyPinnedMessages();
}
void PeerData::addPinnedMessage(MsgId messageId) {
if (messageIdTooSmall(messageId)) {
return;
}
ensurePinnedMessagesCreated();
_pinnedMessages->add(messageId);
}
void PeerData::addPinnedSlice(
std::vector<MsgId> &&ids,
MsgId from,
MsgId till) {
const auto min = [&] {
if (const auto channel = asChannel()) {
return channel->availableMinId();
}
return MsgId(0);
return 0;
}();
messageId = (messageId > min) ? messageId : MsgId(0);
if (_pinnedMessageId != messageId) {
_pinnedMessageId = messageId;
session().changes().peerUpdated(this, UpdateFlag::PinnedMessage);
ids.erase(
ranges::remove_if(ids, [&](MsgId id) { return id <= min; }),
end(ids));
if (from <= min) {
from = 0;
}
if (ids.empty() && !_pinnedMessages) {
return;
}
ensurePinnedMessagesCreated();
_pinnedMessages->add(std::move(ids), from, till, std::nullopt);
}
void PeerData::removePinnedMessage(MsgId messageId) {
if (!_pinnedMessages) {
return;
}
_pinnedMessages->remove(messageId);
removeEmptyPinnedMessages();
}
bool PeerData::canExportChatHistory() const {

View file

@ -29,6 +29,7 @@ class Session;
namespace Data {
class Session;
class PinnedMessages;
int PeerColorIndex(PeerId peerId);
int PeerColorIndex(int32 bareId);
@ -326,13 +327,12 @@ public:
[[nodiscard]] bool canPinMessages() const;
[[nodiscard]] bool canEditMessagesIndefinitely() const;
[[nodiscard]] MsgId pinnedMessageId() const {
return _pinnedMessageId;
}
void setPinnedMessageId(MsgId messageId);
void clearPinnedMessage() {
setPinnedMessageId(0);
}
[[nodiscard]] MsgId topPinnedMessageId() const;
void setTopPinnedMessageId(MsgId messageId);
void clearPinnedMessages(MsgId lessThanId = ServerMaxMsgId);
void addPinnedMessage(MsgId messageId);
void addPinnedSlice(std::vector<MsgId> &&ids, MsgId from, MsgId till);
void removePinnedMessage(MsgId messageId);
[[nodiscard]] bool canExportChatHistory() const;
@ -411,6 +411,10 @@ private:
-> const std::vector<Data::UnavailableReason> &;
void setUserpicChecked(PhotoId photoId, const ImageLocation &location);
void ensurePinnedMessagesCreated();
void removeEmptyPinnedMessages();
[[nodiscard]] bool messageIdTooSmall(MsgId messageId) const;
static constexpr auto kUnknownPhotoId = PhotoId(0xFFFFFFFFFFFFFFFFULL);
@ -428,7 +432,7 @@ private:
base::flat_set<QChar> _nameFirstLetters;
crl::time _lastFullUpdate = 0;
MsgId _pinnedMessageId = 0;
std::unique_ptr<Data::PinnedMessages> _pinnedMessages;
Settings _settings = { kSettingsUnknown };
BlockStatus _blockStatus = BlockStatus::Unknown;

View file

@ -0,0 +1,88 @@
/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "data/data_pinned_messages.h"
namespace Data {
PinnedMessages::PinnedMessages(ChannelId channelId) : _channelId(channelId) {
}
bool PinnedMessages::empty() const {
return _list.empty();
}
MsgId PinnedMessages::topId() const {
const auto slice = _list.snapshot(MessagesQuery{
.aroundId = MaxMessagePosition,
.limitBefore = 1,
.limitAfter = 1
});
return slice.messageIds.empty() ? 0 : slice.messageIds.back().fullId.msg;
}
MessagePosition PinnedMessages::position(MsgId id) const {
return MessagePosition{
.fullId = FullMsgId(_channelId, id),
};
}
void PinnedMessages::add(MsgId messageId) {
_list.addOne(position(messageId));
}
void PinnedMessages::add(
std::vector<MsgId> &&ids,
MsgId from,
MsgId till,
std::optional<int> count) {
auto positions = ids | ranges::view::transform([&](MsgId id) {
return position(id);
}) | ranges::to_vector;
_list.addSlice(
std::move(positions),
MessagesRange{
.from = from ? position(from) : MinMessagePosition,
.till = position(till)
},
count);
}
void PinnedMessages::remove(MsgId messageId) {
_list.removeOne(MessagePosition{
.fullId = FullMsgId(0, messageId),
});
}
void PinnedMessages::setTopId(MsgId messageId) {
while (true) {
auto top = topId();
if (top > messageId) {
remove(top);
} else if (top == messageId) {
return;
} else {
break;
}
}
const auto position = MessagePosition{
.fullId = FullMsgId(0, messageId),
};
_list.addSlice(
{ position },
{ .from = position, .till = MaxMessagePosition },
std::nullopt);
}
void PinnedMessages::clearLessThanId(MsgId messageId) {
_list.removeLessThan(MessagePosition{
.fullId = FullMsgId(0, messageId),
});
}
} // namespace Data

View file

@ -0,0 +1,41 @@
/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "data/data_messages.h"
namespace Data {
class PinnedMessages final {
public:
explicit PinnedMessages(ChannelId channelId);
[[nodiscard]] bool empty() const;
[[nodiscard]] MsgId topId() const;
void add(MsgId messageId);
void add(
std::vector<MsgId> &&ids,
MsgId from,
MsgId till,
std::optional<int> count);
void remove(MsgId messageId);
void setTopId(MsgId messageId);
void clearLessThanId(MsgId messageId);
private:
[[nodiscard]] MessagePosition position(MsgId id) const;
MessagesList _list;
ChannelId _channelId = 0;
};
} // namespace Data

View file

@ -262,9 +262,9 @@ void ApplyUserUpdate(not_null<UserData*> user, const MTPDuserFull &update) {
user->setBotInfoVersion(-1);
}
if (const auto pinned = update.vpinned_msg_id()) {
user->setPinnedMessageId(pinned->v);
user->setTopPinnedMessageId(pinned->v);
} else {
user->clearPinnedMessage();
user->clearPinnedMessages();
}
user->setFullFlags(update.vflags().v);
user->setIsBlocked(update.is_blocked());

View file

@ -177,9 +177,7 @@ void History::itemVanished(not_null<HistoryItem*> item) {
&& unreadCount() > 0) {
setUnreadCount(unreadCount() - 1);
}
if (peer->pinnedMessageId() == item->id) {
peer->clearPinnedMessage();
}
peer->removePinnedMessage(item->id);
}
void History::setLocalDraft(std::unique_ptr<Data::Draft> &&draft) {
@ -709,6 +707,9 @@ not_null<HistoryItem*> History::addNewToBack(
item->id,
{ from, till }));
}
if (item->isPinned()) {
item->history()->peer->addPinnedMessage(item->id);
}
}
if (item->from()->id) {
if (auto user = item->from()->asUser()) {
@ -1005,8 +1006,8 @@ void History::applyServiceChanges(
if (const auto replyTo = data.vreply_to()) {
replyTo->match([&](const MTPDmessageReplyHeader &data) {
if (item) {
item->history()->peer->setPinnedMessageId(
data.vreply_to_msg_id().v);
//item->history()->peer->setTopPinnedMessageId( // #TODO pinned
// data.vreply_to_msg_id().v);
}
});
}
@ -1160,6 +1161,7 @@ void History::addEdgesToSharedMedia() {
{},
{ from, till }));
}
peer->addPinnedSlice({}, from, till);
}
void History::addOlderSlice(const QVector<MTPMessage> &slice) {
@ -1323,6 +1325,7 @@ void History::checkAddAllToUnreadMentions() {
void History::addToSharedMedia(
const std::vector<not_null<HistoryItem*>> &items) {
auto pinned = std::vector<MsgId>();
std::vector<MsgId> medias[Storage::kSharedMediaTypeCount];
for (const auto item : items) {
if (const auto types = item->sharedMediaTypes()) {
@ -1336,6 +1339,12 @@ void History::addToSharedMedia(
}
}
}
if (item->isPinned()) {
if (pinned.empty()) {
pinned.reserve(items.size());
}
pinned.push_back(item->id);
}
}
const auto from = loadedAtTop() ? 0 : minMsgId();
const auto till = loadedAtBottom() ? ServerMaxMsgId : maxMsgId();
@ -1349,6 +1358,7 @@ void History::addToSharedMedia(
{ from, till }));
}
}
peer->addPinnedSlice(std::move(pinned), from, till);
}
void History::calculateFirstUnreadMessage() {
@ -3017,7 +3027,7 @@ void History::clear(ClearType type) {
clearSharedMedia();
clearLastKeyboard();
if (const auto channel = peer->asChannel()) {
channel->clearPinnedMessage();
channel->clearPinnedMessages();
//if (const auto feed = channel->feed()) { // #feed
// // Should be after resetting the _lastMessage.
// feed->historyCleared(this);

View file

@ -347,6 +347,16 @@ void HistoryItem::markMediaRead() {
}
}
void HistoryItem::setIsPinned(bool pinned) {
if (pinned) {
_flags |= MTPDmessage::Flag::f_pinned;
history()->peer->addPinnedMessage(id);
} else {
_flags &= ~MTPDmessage::Flag::f_pinned;
history()->peer->removePinnedMessage(id);
}
}
bool HistoryItem::definesReplyKeyboard() const {
if (const auto markup = Get<HistoryMessageReplyMarkup>()) {
if (markup->flags & MTPDreplyKeyboardMarkup_ClientFlag::f_inline) {
@ -476,6 +486,9 @@ void HistoryItem::indexAsNewItem() {
types,
id));
}
if (isPinned()) {
_history->peer->setTopPinnedMessageId(id);
}
//if (const auto channel = history()->peer->asChannel()) { // #feed
// if (const auto feed = channel->feed()) {
// _history->session().storage().add(Storage::FeedMessagesAddNew(
@ -509,10 +522,6 @@ void HistoryItem::setRealId(MsgId newId) {
_history->owner().requestItemRepaint(this);
}
bool HistoryItem::isPinned() const {
return (_history->peer->pinnedMessageId() == id);
}
bool HistoryItem::canPin() const {
if (id < 0 || !toHistoryMessage()) {
return false;
@ -656,7 +665,7 @@ ChannelId HistoryItem::channelId() const {
}
Data::MessagePosition HistoryItem::position() const {
return Data::MessagePosition(date(), fullId());
return { .fullId = fullId(), .date = date() };
}
MsgId HistoryItem::replyToId() const {

View file

@ -113,6 +113,9 @@ public:
[[nodiscard]] bool out() const {
return _flags & MTPDmessage::Flag::f_out;
}
[[nodiscard]] bool isPinned() const {
return _flags & MTPDmessage::Flag::f_pinned;
}
[[nodiscard]] bool unread() const;
[[nodiscard]] bool showNotification() const;
void markClientSideAsRead();
@ -121,6 +124,7 @@ public:
[[nodiscard]] bool isUnreadMedia() const;
[[nodiscard]] bool hasUnreadMediaFlag() const;
void markMediaRead();
void setIsPinned(bool isPinned);
// For edit media in history_message.
virtual void returnSavedMedia() {};
@ -312,7 +316,6 @@ public:
return _text.isEmpty();
}
[[nodiscard]] bool isPinned() const;
[[nodiscard]] bool canPin() const;
[[nodiscard]] bool canStopPoll() const;
[[nodiscard]] virtual bool allowsSendNow() const;

View file

@ -5200,6 +5200,7 @@ void HistoryWidget::updatePinnedBar(bool force) {
if (!_pinnedBar) {
return;
}
const auto messageId = _pinnedBar->msgId;
if (!force) {
if (_pinnedBar->msg) {
return;
@ -5208,7 +5209,9 @@ void HistoryWidget::updatePinnedBar(bool force) {
Assert(_history != nullptr);
if (!_pinnedBar->msg) {
_pinnedBar->msg = session().data().message(_history->channelId(), _pinnedBar->msgId);
_pinnedBar->msg = session().data().message(
_history->channelId(),
messageId);
}
if (_pinnedBar->msg) {
_pinnedBar->text.setText(
@ -5217,17 +5220,15 @@ void HistoryWidget::updatePinnedBar(bool force) {
Ui::DialogTextOptions());
update();
} else if (force) {
if (auto channel = _peer ? _peer->asChannel() : nullptr) {
channel->clearPinnedMessage();
}
destroyPinnedBar();
_history->peer->removePinnedMessage(messageId);
updateControlsGeometry();
}
}
bool HistoryWidget::pinnedMsgVisibilityUpdated() {
auto result = false;
auto pinnedId = _peer->pinnedMessageId();
auto pinnedId = _peer->topPinnedMessageId();
if (pinnedId && !_peer->canPinMessages()) {
const auto hiddenId = session().settings().hiddenPinnedMessageId(
_peer->id);
@ -5540,23 +5541,23 @@ void HistoryWidget::unpinMessage(FullMsgId itemId) {
if (!_peer) {
return;
}
UnpinMessage(_peer);
UnpinMessage(_peer, itemId.msg);
}
void HistoryWidget::UnpinMessage(not_null<PeerData*> peer) {
void HistoryWidget::UnpinMessage(not_null<PeerData*> peer, MsgId msgId) {
if (!peer) {
return;
}
const auto session = &peer->session();
Ui::show(Box<ConfirmBox>(tr::lng_pinned_unpin_sure(tr::now), tr::lng_pinned_unpin(tr::now), crl::guard(session, [=] {
peer->clearPinnedMessage();
peer->removePinnedMessage(msgId);
Ui::hideLayer();
session->api().request(MTPmessages_UpdatePinnedMessage(
MTP_flags(0),
MTP_flags(MTPmessages_UpdatePinnedMessage::Flag::f_unpin),
peer->input,
MTP_int(0)
MTP_int(msgId)
)).done([=](const MTPUpdates &result) {
session->api().applyUpdates(result);
}).send();
@ -5564,7 +5565,7 @@ void HistoryWidget::UnpinMessage(not_null<PeerData*> peer) {
}
void HistoryWidget::hidePinnedMessage() {
const auto pinnedId = _peer ? _peer->pinnedMessageId() : MsgId(0);
const auto pinnedId = _peer ? _peer->topPinnedMessageId() : MsgId(0);
if (!pinnedId) {
if (pinnedMsgVisibilityUpdated()) {
updateControlsGeometry();

View file

@ -530,7 +530,7 @@ private:
void addMessagesToFront(PeerData *peer, const QVector<MTPMessage> &messages);
void addMessagesToBack(PeerData *peer, const QVector<MTPMessage> &messages);
static void UnpinMessage(not_null<PeerData*> peer);
static void UnpinMessage(not_null<PeerData*> peer, MsgId msgId);
void updateHistoryGeometry(bool initial = false, bool loadedDown = false, const ScrollChange &change = { ScrollChangeNone, 0 });
void updateListSize();

View file

@ -85,8 +85,9 @@ RepliesMemento::RepliesMemento(
: RepliesMemento(commentsItem->history(), commentsItem->id, commentId) {
if (commentId) {
_list.setAroundPosition({
TimeId(0),
FullMsgId(commentsItem->history()->channelId(), commentId)
.fullId = FullMsgId(commentsItem->history()->channelId(), commentId),
.date = TimeId(0),
});
} else if (commentsItem->computeRepliesInboxReadTillFull() == MsgId(1)) {
_list.setAroundPosition(Data::MinMessagePosition);
@ -1430,7 +1431,9 @@ bool RepliesWidget::showMessage(
}
return nullptr;
}();
showAtPosition(Data::MessagePosition(message->date(), id), originItem);
showAtPosition(
Data::MessagePosition{ .fullId = id, .date = message->date() },
originItem);
return true;
}
@ -1479,8 +1482,8 @@ void RepliesWidget::restoreState(not_null<RepliesMemento*> memento) {
_inner->restoreState(memento->list());
if (const auto highlight = memento->getHighlightId()) {
const auto position = Data::MessagePosition{
TimeId(0),
FullMsgId(_history->channelId(), highlight)
.fullId = FullMsgId(_history->channelId(), highlight),
.date = TimeId(0),
};
_inner->showAroundPosition(position, [=] {
return showAtPositionNow(position, nullptr);