kotatogram-desktop/Telegram/SourceFiles/history/history.cpp

3633 lines
96 KiB
C++

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "history/history.h"
#include "history/view/history_view_element.h"
#include "history/view/history_view_item_preview.h"
#include "history/view/history_view_translate_tracker.h"
#include "dialogs/dialogs_indexed_list.h"
#include "history/history_inner_widget.h"
#include "history/history_item.h"
#include "history/history_item_components.h"
#include "history/history_item_helpers.h"
#include "history/history_translation.h"
#include "history/history_unread_things.h"
#include "dialogs/ui/dialogs_layout.h"
#include "data/business/data_shortcut_messages.h"
#include "data/components/scheduled_messages.h"
#include "data/components/sponsored_messages.h"
#include "data/components/top_peers.h"
#include "data/notify/data_notify_settings.h"
#include "data/stickers/data_stickers.h"
#include "data/data_drafts.h"
#include "data/data_saved_sublist.h"
#include "data/data_session.h"
#include "data/data_media_types.h"
#include "data/data_channel_admins.h"
#include "data/data_changes.h"
#include "data/data_chat_filters.h"
#include "data/data_send_action.h"
#include "data/data_folder.h"
#include "data/data_forum.h"
#include "data/data_forum_topic.h"
#include "data/data_photo.h"
#include "data/data_channel.h"
#include "data/data_chat.h"
#include "data/data_user.h"
#include "data/data_document.h"
#include "data/data_histories.h"
#include "data/data_history_messages.h"
#include "lang/lang_keys.h"
#include "apiwrap.h"
#include "api/api_chat_participants.h"
#include "mainwidget.h"
#include "mainwindow.h"
#include "main/main_session.h"
#include "window/notifications_manager.h"
#include "calls/calls_instance.h"
#include "spellcheck/spellcheck_types.h"
#include "storage/localstorage.h"
#include "storage/storage_facade.h"
#include "storage/storage_shared_media.h"
#include "storage/storage_account.h"
#include "support/support_helper.h"
#include "ui/image/image.h"
#include "ui/text/text_options.h"
#include "ui/text/text_utilities.h"
#include "ui/toast/toast.h"
#include "payments/payments_checkout_process.h"
#include "core/crash_reports.h"
#include "core/application.h"
#include "base/unixtime.h"
#include "base/qt/qt_common_adapters.h"
#include "styles/style_dialogs.h"
namespace {
constexpr auto kNewBlockEachMessage = 50;
constexpr auto kSkipCloudDraftsFor = TimeId(2);
using UpdateFlag = Data::HistoryUpdate::Flag;
[[nodiscard]] HistoryItemCommonFields WithLocalFlag(
HistoryItemCommonFields fields) {
fields.flags |= MessageFlag::Local;
return fields;
}
} // namespace
History::History(not_null<Data::Session*> owner, PeerId peerId)
: Thread(owner, Type::History)
, peer(owner->peer(peerId))
, _delegateMixin(HistoryInner::DelegateMixin())
, _chatListNameSortKey(owner->nameSortKey(peer->name()))
, _sendActionPainter(this) {
Thread::setMuted(owner->notifySettings().isMuted(peer));
if (const auto user = peer->asUser()) {
if (user->isBot()) {
_outboxReadBefore = std::numeric_limits<MsgId>::max();
}
}
}
History::~History() = default;
void History::clearLastKeyboard() {
if (lastKeyboardId) {
if (lastKeyboardId == lastKeyboardHiddenId) {
lastKeyboardHiddenId = 0;
}
lastKeyboardId = 0;
session().changes().historyUpdated(this, UpdateFlag::BotKeyboard);
}
lastKeyboardInited = true;
lastKeyboardFrom = 0;
}
int History::height() const {
return _height;
}
bool History::hasPendingResizedItems() const {
return _flags & Flag::HasPendingResizedItems;
}
void History::setHasPendingResizedItems() {
_flags |= Flag::HasPendingResizedItems;
}
void History::itemRemoved(not_null<HistoryItem*> item) {
if (item == _joinedMessage) {
_joinedMessage = nullptr;
}
item->removeMainView();
if (_lastServerMessage == item) {
_lastServerMessage = std::nullopt;
}
if (lastMessage() == item) {
_lastMessage = std::nullopt;
if (loadedAtBottom()) {
if (const auto last = lastAvailableMessage()) {
setLastMessage(last);
}
}
}
checkChatListMessageRemoved(item);
itemVanished(item);
if (IsClientMsgId(item->id)) {
unregisterClientSideMessage(item);
}
if (const auto topic = item->topic()) {
topic->applyItemRemoved(item->id);
}
if (const auto chat = peer->asChat()) {
if (const auto to = chat->getMigrateToChannel()) {
if (const auto history = owner().historyLoaded(to)) {
history->checkChatListMessageRemoved(item);
}
}
}
}
void History::checkChatListMessageRemoved(not_null<HistoryItem*> item) {
if (chatListMessage() != item) {
return;
}
setChatListMessageUnknown();
refreshChatListMessage();
}
void History::itemVanished(not_null<HistoryItem*> item) {
item->notificationThread()->removeNotification(item);
if (lastKeyboardId == item->id) {
clearLastKeyboard();
}
if ((!item->out() || item->isPost())
&& item->unread(this)
&& unreadCount() > 0) {
setUnreadCount(unreadCount() - 1);
}
}
void History::takeLocalDraft(not_null<History*> from) {
const auto topicRootId = MsgId(0);
const auto i = from->_drafts.find(Data::DraftKey::Local(topicRootId));
if (i == end(from->_drafts)) {
return;
}
auto &draft = i->second;
if (!draft->textWithTags.text.isEmpty()
&& !_drafts.contains(Data::DraftKey::Local(topicRootId))) {
// Edit and reply to drafts can't migrate.
// Cloud drafts do not migrate automatically.
draft->reply = FullReplyTo();
setLocalDraft(std::move(draft));
}
from->clearLocalDraft(topicRootId);
session().api().saveDraftToCloudDelayed(from);
}
void History::createLocalDraftFromCloud(MsgId topicRootId) {
const auto draft = cloudDraft(topicRootId);
if (!draft) {
clearLocalDraft(topicRootId);
return;
} else if (Data::DraftIsNull(draft) || !draft->date) {
return;
}
draft->reply.topicRootId = topicRootId;
auto existing = localDraft(topicRootId);
if (Data::DraftIsNull(existing)
|| !existing->date
|| draft->date >= existing->date) {
if (!existing) {
setLocalDraft(std::make_unique<Data::Draft>(
draft->textWithTags,
draft->reply,
draft->cursor,
draft->webpage));
existing = localDraft(topicRootId);
} else if (existing != draft) {
existing->textWithTags = draft->textWithTags;
existing->reply = draft->reply;
existing->cursor = draft->cursor;
existing->webpage = draft->webpage;
}
existing->date = draft->date;
}
}
Data::Draft *History::draft(Data::DraftKey key) const {
if (!key) {
return nullptr;
}
const auto i = _drafts.find(key);
return (i != _drafts.end()) ? i->second.get() : nullptr;
}
void History::setDraft(
Data::DraftKey key,
std::unique_ptr<Data::Draft> &&draft) {
if (!key) {
return;
}
const auto cloudThread = key.isCloud()
? threadFor(key.topicRootId())
: nullptr;
if (cloudThread) {
cloudThread->cloudDraftTextCache().clear();
}
if (draft) {
_drafts[key] = std::move(draft);
} else if (_drafts.remove(key) && cloudThread) {
cloudThread->updateChatListSortPosition();
}
}
const Data::HistoryDrafts &History::draftsMap() const {
return _drafts;
}
void History::setDraftsMap(Data::HistoryDrafts &&map) {
for (auto &[key, draft] : _drafts) {
map[key] = std::move(draft);
}
_drafts = std::move(map);
}
void History::clearDraft(Data::DraftKey key) {
setDraft(key, nullptr);
}
void History::clearDrafts() {
for (auto &[key, draft] : base::take(_drafts)) {
const auto cloudThread = key.isCloud()
? threadFor(key.topicRootId())
: nullptr;
if (cloudThread) {
cloudThread->cloudDraftTextCache().clear();
cloudThread->updateChatListSortPosition();
}
}
}
Data::Draft *History::createCloudDraft(
MsgId topicRootId,
const Data::Draft *fromDraft) {
if (Data::DraftIsNull(fromDraft)) {
setCloudDraft(std::make_unique<Data::Draft>(
TextWithTags(),
FullReplyTo{ .topicRootId = topicRootId },
MessageCursor(),
Data::WebPageDraft()));
cloudDraft(topicRootId)->date = TimeId(0);
} else {
auto existing = cloudDraft(topicRootId);
if (!existing) {
auto reply = fromDraft->reply;
reply.topicRootId = topicRootId;
setCloudDraft(std::make_unique<Data::Draft>(
fromDraft->textWithTags,
reply,
fromDraft->cursor,
fromDraft->webpage));
existing = cloudDraft(topicRootId);
} else if (existing != fromDraft) {
existing->textWithTags = fromDraft->textWithTags;
existing->reply = fromDraft->reply;
existing->cursor = fromDraft->cursor;
existing->webpage = fromDraft->webpage;
}
existing->date = base::unixtime::now();
existing->reply.topicRootId = topicRootId;
}
if (const auto thread = threadFor(topicRootId)) {
thread->cloudDraftTextCache().clear();
thread->updateChatListSortPosition();
}
return cloudDraft(topicRootId);
}
bool History::skipCloudDraftUpdate(MsgId topicRootId, TimeId date) const {
const auto i = _acceptCloudDraftsAfter.find(topicRootId);
return _savingCloudDraftRequests.contains(topicRootId)
|| (i != _acceptCloudDraftsAfter.end() && date < i->second);
}
void History::startSavingCloudDraft(MsgId topicRootId) {
++_savingCloudDraftRequests[topicRootId];
}
void History::finishSavingCloudDraft(MsgId topicRootId, TimeId savedAt) {
const auto i = _savingCloudDraftRequests.find(topicRootId);
if (i != _savingCloudDraftRequests.end()) {
if (--i->second <= 0) {
_savingCloudDraftRequests.erase(i);
}
}
auto &after = _acceptCloudDraftsAfter[topicRootId];
after = std::max(after, savedAt + kSkipCloudDraftsFor);
}
void History::applyCloudDraft(MsgId topicRootId) {
if (!topicRootId && session().supportMode()) {
updateChatListEntry();
session().supportHelper().cloudDraftChanged(this);
} else {
createLocalDraftFromCloud(topicRootId);
if (const auto thread = threadFor(topicRootId)) {
thread->updateChatListSortPosition();
if (!topicRootId) {
session().changes().historyUpdated(
this,
UpdateFlag::CloudDraft);
} else {
session().changes().topicUpdated(
thread->asTopic(),
Data::TopicUpdate::Flag::CloudDraft);
}
}
}
}
void History::draftSavedToCloud(MsgId topicRootId) {
if (const auto thread = threadFor(topicRootId)) {
thread->updateChatListEntry();
}
session().local().writeDrafts(this);
}
const Data::ForwardDraft &History::forwardDraft(
MsgId topicRootId) const {
static const auto kEmpty = Data::ForwardDraft();
const auto i = _forwardDrafts.find(topicRootId);
return (i != end(_forwardDrafts)) ? i->second : kEmpty;
}
Data::ResolvedForwardDraft History::resolveForwardDraft(
const Data::ForwardDraft &draft) const {
return Data::ResolvedForwardDraft{
.items = owner().idsToItems(draft.ids),
.options = draft.options,
.groupOptions = draft.groupOptions,
};
}
Data::ResolvedForwardDraft History::resolveForwardDraft(
MsgId topicRootId) {
const auto &draft = forwardDraft(topicRootId);
auto result = resolveForwardDraft(draft);
if (result.items.size() != draft.ids.size()) {
setForwardDraft(topicRootId, {
.ids = owner().itemsToIds(result.items),
.options = result.options,
.groupOptions = result.groupOptions,
});
}
return result;
}
void History::setForwardDraft(
MsgId topicRootId,
Data::ForwardDraft &&draft) {
auto changed = false;
if (draft.ids.empty()) {
changed = _forwardDrafts.remove(topicRootId);
} else {
auto &now = _forwardDrafts[topicRootId];
if (now != draft) {
now = std::move(draft);
changed = true;
}
}
if (changed) {
const auto entry = topicRootId
? peer->forumTopicFor(topicRootId)
: (Dialogs::Entry*)this;
if (entry) {
session().changes().entryUpdated(
entry,
Data::EntryUpdate::Flag::ForwardDraft);
}
}
}
not_null<HistoryItem*> History::createItem(
MsgId id,
const MTPMessage &message,
MessageFlags localFlags,
bool detachExistingItem,
bool newMessage) {
if (const auto result = owner().message(peer, id)) {
if (detachExistingItem) {
result->removeMainView();
}
return result;
}
const auto result = message.match([&](const auto &data) {
return makeMessage(id, data, localFlags);
});
if (newMessage && result->out() && result->isRegular()) {
session().topPeers().increment(peer, result->date());
}
return result;
}
std::vector<not_null<HistoryItem*>> History::createItems(
const QVector<MTPMessage> &data) {
auto result = std::vector<not_null<HistoryItem*>>();
result.reserve(data.size());
const auto localFlags = MessageFlags();
const auto detachExistingItem = true;
for (auto i = data.cend(), e = data.cbegin(); i != e;) {
const auto &data = *--i;
result.emplace_back(createItem(
IdFromMessage(data),
data,
localFlags,
detachExistingItem));
}
return result;
}
not_null<HistoryItem*> History::addNewMessage(
MsgId id,
const MTPMessage &message,
MessageFlags localFlags,
NewMessageType type) {
const auto newMessage = (type == NewMessageType::Unread);
const auto detachExisting = newMessage;
const auto item = createItem(
id,
message,
localFlags,
detachExisting,
newMessage);
if (type == NewMessageType::Existing || item->mainView()) {
return item;
}
if (newMessage && item->isHistoryEntry()) {
applyMessageChanges(item, message);
}
return addNewItem(item, newMessage);
}
not_null<HistoryItem*> History::insertItem(
std::unique_ptr<HistoryItem> item) {
Expects(item != nullptr);
const auto &[i, ok] = _items.insert(std::move(item));
const auto result = i->get();
owner().registerMessage(result);
Ensures(ok);
return result;
}
void History::destroyMessage(not_null<HistoryItem*> item) {
Expects(item->isHistoryEntry() || !item->mainView());
const auto peerId = peer->id;
if (item->isHistoryEntry()) {
// All this must be done for all items manually in History::clear()!
item->destroyHistoryEntry();
if (item->isRegular()) {
if (const auto messages = _messages.get()) {
messages->removeOne(item->id);
}
if (const auto types = item->sharedMediaTypes()) {
session().storage().remove(Storage::SharedMediaRemoveOne(
peerId,
types,
item->id));
}
}
itemRemoved(item);
}
if (item->isSending()) {
session().api().cancelLocalItem(item);
}
const auto documentToCancel = [&] {
const auto media = item->isAdminLogEntry()
? nullptr
: item->media();
return media ? media->document() : nullptr;
}();
owner().unregisterMessage(item);
Core::App().notifications().clearFromItem(item);
auto hack = std::unique_ptr<HistoryItem>(item.get());
const auto i = _items.find(hack);
hack.release();
Assert(i != end(_items));
_items.erase(i);
if (documentToCancel) {
session().data().documentMessageRemoved(documentToCancel);
}
}
void History::destroyMessagesByDates(TimeId minDate, TimeId maxDate) {
auto toDestroy = std::vector<not_null<HistoryItem*>>();
toDestroy.reserve(_items.size());
for (const auto &message : _items) {
if (message->isRegular()
&& message->date() > minDate
&& message->date() < maxDate) {
toDestroy.push_back(message.get());
}
}
for (const auto item : toDestroy) {
item->destroy();
}
}
void History::destroyMessagesByTopic(MsgId topicRootId) {
auto toDestroy = std::vector<not_null<HistoryItem*>>();
toDestroy.reserve(_items.size());
for (const auto &message : _items) {
if (message->topicRootId() == topicRootId) {
toDestroy.push_back(message.get());
}
}
for (const auto item : toDestroy) {
item->destroy();
}
}
void History::unpinMessagesFor(MsgId topicRootId) {
if (!topicRootId) {
session().storage().remove(
Storage::SharedMediaRemoveAll(
peer->id,
Storage::SharedMediaType::Pinned));
setHasPinnedMessages(false);
if (const auto forum = peer->forum()) {
forum->enumerateTopics([](not_null<Data::ForumTopic*> topic) {
topic->setHasPinnedMessages(false);
});
}
for (const auto &item : _items) {
if (item->isPinned()) {
item->setIsPinned(false);
}
}
} else {
session().storage().remove(
Storage::SharedMediaRemoveAll(
peer->id,
topicRootId,
Storage::SharedMediaType::Pinned));
if (const auto topic = peer->forumTopicFor(topicRootId)) {
topic->setHasPinnedMessages(false);
}
for (const auto &item : _items) {
if (item->isPinned() && item->topicRootId() == topicRootId) {
item->setIsPinned(false);
}
}
}
}
not_null<HistoryItem*> History::addNewItem(
not_null<HistoryItem*> item,
bool unread) {
if (item->isScheduled()) {
session().scheduledMessages().appendSending(item);
return item;
} else if (item->isBusinessShortcut()) {
owner().shortcutMessages().appendSending(item);
return item;
} else if (!item->isHistoryEntry()) {
return item;
}
// In case we've loaded a new 'last' message
// and it is not in blocks and we think that
// we have all the messages till the bottom
// we should unload known history or mark
// currently loaded slice as not reaching bottom.
const auto shouldMarkBottomNotLoaded = loadedAtBottom()
&& !unread
&& !isEmpty();
if (shouldMarkBottomNotLoaded) {
setNotLoadedAtBottom();
}
if (!loadedAtBottom() || peer->migrateTo()) {
setLastMessage(item);
if (unread) {
newItemAdded(item);
}
} else {
addNewToBack(item, unread);
checkForLoadedAtTop(item);
}
if (const auto sublist = item->savedSublist()) {
sublist->applyMaybeLast(item, unread);
}
return item;
}
void History::checkForLoadedAtTop(not_null<HistoryItem*> added) {
if (peer->isChat()) {
if (added->isGroupEssential() && !added->isGroupMigrate()) {
// We added the first message about group creation.
_loadedAtTop = true;
addEdgesToSharedMedia();
}
} else if (peer->isChannel()) {
if (added->id == 1) {
_loadedAtTop = true;
checkLocalMessages();
addEdgesToSharedMedia();
}
}
}
not_null<HistoryItem*> History::addNewLocalMessage(
HistoryItemCommonFields &&fields,
const TextWithEntities &text,
const MTPMessageMedia &media) {
return addNewItem(
makeMessage(WithLocalFlag(std::move(fields)), text, media),
true);
}
not_null<HistoryItem*> History::addNewLocalMessage(
HistoryItemCommonFields &&fields,
not_null<HistoryItem*> forwardOriginal) {
return addNewItem(
makeMessage(WithLocalFlag(std::move(fields)), forwardOriginal),
true);
}
not_null<HistoryItem*> History::addNewLocalMessage(
HistoryItemCommonFields &&fields,
not_null<DocumentData*> document,
const TextWithEntities &caption) {
return addNewItem(
makeMessage(WithLocalFlag(std::move(fields)), document, caption),
true);
}
not_null<HistoryItem*> History::addNewLocalMessage(
HistoryItemCommonFields &&fields,
not_null<PhotoData*> photo,
const TextWithEntities &caption) {
return addNewItem(
makeMessage(WithLocalFlag(std::move(fields)), photo, caption),
true);
}
not_null<HistoryItem*> History::addNewLocalMessage(
HistoryItemCommonFields &&fields,
not_null<GameData*> game) {
return addNewItem(
makeMessage(WithLocalFlag(std::move(fields)), game),
true);
}
not_null<HistoryItem*> History::addSponsoredMessage(
MsgId id,
Data::SponsoredFrom from,
const TextWithEntities &textWithEntities) {
return addNewItem(
makeMessage(id, from, textWithEntities, nullptr),
true);
}
void History::clearUnreadMentionsFor(MsgId topicRootId) {
const auto forum = peer->forum();
if (!topicRootId) {
if (forum) {
forum->clearAllUnreadMentions();
}
unreadMentions().clear();
return;
} else if (forum) {
if (const auto topic = forum->topicFor(topicRootId)) {
topic->unreadMentions().clear();
}
}
const auto &ids = unreadMentionsIds();
if (ids.empty()) {
return;
}
const auto owner = &this->owner();
const auto peerId = peer->id;
auto items = base::flat_set<MsgId>();
items.reserve(ids.size());
for (const auto &id : ids) {
if (const auto item = owner->message(peerId, id)) {
if (item->topicRootId() == topicRootId) {
items.emplace(id);
}
}
}
for (const auto &id : items) {
unreadMentions().erase(id);
}
}
void History::clearUnreadReactionsFor(MsgId topicRootId) {
const auto forum = peer->forum();
if (!topicRootId) {
if (forum) {
forum->clearAllUnreadReactions();
}
unreadReactions().clear();
return;
} else if (forum) {
if (const auto topic = forum->topicFor(topicRootId)) {
topic->unreadReactions().clear();
}
}
const auto &ids = unreadReactionsIds();
if (ids.empty()) {
return;
}
const auto owner = &this->owner();
const auto peerId = peer->id;
auto items = base::flat_set<MsgId>();
items.reserve(ids.size());
for (const auto &id : ids) {
if (const auto item = owner->message(peerId, id)) {
if (item->topicRootId() == topicRootId) {
items.emplace(id);
}
}
}
for (const auto &id : items) {
unreadReactions().erase(id);
}
}
not_null<HistoryItem*> History::addNewToBack(
not_null<HistoryItem*> item,
bool unread) {
Expects(!isBuildingFrontBlock());
addItemToBlock(item);
if (!unread && item->isRegular()) {
const auto from = loadedAtTop() ? 0 : minMsgId();
const auto till = loadedAtBottom() ? ServerMaxMsgId : maxMsgId();
if (_messages) {
_messages->addExisting(item->id, { from, till });
}
if (const auto types = item->sharedMediaTypes()) {
auto &storage = session().storage();
storage.add(Storage::SharedMediaAddExisting(
peer->id,
MsgId(0), // topicRootId
types,
item->id,
{ from, till }));
const auto pinned = types.test(Storage::SharedMediaType::Pinned);
if (pinned) {
setHasPinnedMessages(true);
}
if (const auto topic = item->topic()) {
storage.add(Storage::SharedMediaAddExisting(
peer->id,
topic->rootId(),
types,
item->id,
{ item->id, item->id}));
if (pinned) {
topic->setHasPinnedMessages(true);
}
}
}
}
if (item->from()->id) {
if (auto user = item->from()->asUser()) {
auto getLastAuthors = [this]() -> std::deque<not_null<UserData*>>* {
if (auto chat = peer->asChat()) {
return &chat->lastAuthors;
} else if (auto channel = peer->asMegagroup()) {
return channel->canViewMembers()
? &channel->mgInfo->lastParticipants
: nullptr;
}
return nullptr;
};
if (auto megagroup = peer->asMegagroup()) {
if (user->isBot()) {
auto mgInfo = megagroup->mgInfo.get();
Assert(mgInfo != nullptr);
mgInfo->bots.insert(user);
if (mgInfo->botStatus != 0 && mgInfo->botStatus < 2) {
mgInfo->botStatus = 2;
}
}
}
if (auto lastAuthors = getLastAuthors()) {
auto prev = ranges::find(
*lastAuthors,
user,
[](not_null<UserData*> user) { return user.get(); });
auto index = (prev != lastAuthors->end())
? (lastAuthors->end() - prev)
: -1;
if (index > 0) {
lastAuthors->erase(prev);
} else if (index < 0 && peer->isMegagroup()) { // nothing is outdated if just reordering
// admins information outdated
}
if (index) {
lastAuthors->push_front(user);
}
if (auto megagroup = peer->asMegagroup()) {
session().changes().peerUpdated(
peer,
Data::PeerUpdate::Flag::Members);
owner().addNewMegagroupParticipant(megagroup, user);
}
}
}
if (item->definesReplyKeyboard()) {
auto markupFlags = item->replyKeyboardFlags();
if (!(markupFlags & ReplyMarkupFlag::Selective)
|| item->mentionsMe()) {
auto getMarkupSenders = [this]() -> base::flat_set<not_null<PeerData*>>* {
if (auto chat = peer->asChat()) {
return &chat->markupSenders;
} else if (auto channel = peer->asMegagroup()) {
return &channel->mgInfo->markupSenders;
}
return nullptr;
};
if (auto markupSenders = getMarkupSenders()) {
markupSenders->insert(item->from());
}
if (markupFlags & ReplyMarkupFlag::None) {
// None markup means replyKeyboardHide.
if (lastKeyboardFrom == item->from()->id
|| (!lastKeyboardInited
&& !peer->isChat()
&& !peer->isMegagroup()
&& !item->out())) {
clearLastKeyboard();
}
} else {
bool botNotInChat = false;
if (peer->isChat()) {
botNotInChat = item->from()->isUser()
&& (!peer->asChat()->participants.empty()
|| !Data::CanSendAnything(peer))
&& !peer->asChat()->participants.contains(
item->from()->asUser());
} else if (peer->isMegagroup()) {
botNotInChat = item->from()->isUser()
&& (peer->asChannel()->mgInfo->botStatus != 0
|| !Data::CanSendAnything(peer))
&& !peer->asChannel()->mgInfo->bots.contains(
item->from()->asUser());
}
if (botNotInChat) {
clearLastKeyboard();
} else {
lastKeyboardInited = true;
lastKeyboardId = item->id;
lastKeyboardFrom = item->from()->id;
lastKeyboardUsed = false;
}
}
}
}
}
setLastMessage(item);
if (unread) {
newItemAdded(item);
}
owner().notifyHistoryChangeDelayed(this);
return item;
}
void History::applyMessageChanges(
not_null<HistoryItem*> item,
const MTPMessage &data) {
if (data.type() == mtpc_messageService) {
applyServiceChanges(item, data.c_messageService());
}
owner().stickers().checkSavedGif(item);
session().changes().messageUpdated(
item,
Data::MessageUpdate::Flag::NewAdded);
}
void History::applyServiceChanges(
not_null<HistoryItem*> item,
const MTPDmessageService &data) {
const auto replyTo = data.vreply_to();
const auto processJoinedUser = [&](
not_null<ChannelData*> megagroup,
not_null<MegagroupInfo*> mgInfo,
not_null<UserData*> user) {
if (!base::contains(mgInfo->lastParticipants, user)
&& megagroup->canViewMembers()) {
mgInfo->lastParticipants.push_front(user);
session().changes().peerUpdated(
peer,
Data::PeerUpdate::Flag::Members);
owner().addNewMegagroupParticipant(megagroup, user);
}
if (user->isBot()) {
mgInfo->bots.insert(user);
if (mgInfo->botStatus != 0 && mgInfo->botStatus < 2) {
mgInfo->botStatus = 2;
}
}
};
const auto processJoinedPeer = [&](not_null<PeerData*> joined) {
if (const auto megagroup = peer->asMegagroup()) {
const auto mgInfo = megagroup->mgInfo.get();
Assert(mgInfo != nullptr);
if (const auto user = joined->asUser()) {
processJoinedUser(megagroup, mgInfo, user);
}
}
};
data.vaction().match([&](const MTPDmessageActionChatAddUser &data) {
if (const auto megagroup = peer->asMegagroup()) {
const auto mgInfo = megagroup->mgInfo.get();
Assert(mgInfo != nullptr);
for (const auto &userId : data.vusers().v) {
if (const auto user = owner().userLoaded(userId.v)) {
processJoinedUser(megagroup, mgInfo, user);
}
}
}
}, [&](const MTPDmessageActionChatJoinedByLink &data) {
processJoinedPeer(item->from());
}, [&](const MTPDmessageActionChatDeletePhoto &data) {
if (const auto chat = peer->asChat()) {
chat->setPhoto(MTP_chatPhotoEmpty());
}
}, [&](const MTPDmessageActionChatDeleteUser &data) {
const auto uid = data.vuser_id().v;
if (lastKeyboardFrom == peerFromUser(uid)) {
clearLastKeyboard();
}
if (const auto megagroup = peer->asMegagroup()) {
if (const auto user = owner().userLoaded(uid)) {
const auto mgInfo = megagroup->mgInfo.get();
Assert(mgInfo != nullptr);
const auto i = ranges::find(
mgInfo->lastParticipants,
user,
[](not_null<UserData*> user) { return user.get(); });
if (i != mgInfo->lastParticipants.end()) {
mgInfo->lastParticipants.erase(i);
session().changes().peerUpdated(
peer,
Data::PeerUpdate::Flag::Members);
}
owner().removeMegagroupParticipant(megagroup, user);
if (megagroup->membersCount() > 1) {
megagroup->setMembersCount(
megagroup->membersCount() - 1);
} else {
mgInfo->lastParticipantsStatus
|= MegagroupInfo::LastParticipantsCountOutdated;
mgInfo->lastParticipantsCount = 0;
}
if (mgInfo->lastAdmins.contains(user)) {
mgInfo->lastAdmins.remove(user);
if (megagroup->adminsCount() > 1) {
megagroup->setAdminsCount(
megagroup->adminsCount() - 1);
}
session().changes().peerUpdated(
peer,
Data::PeerUpdate::Flag::Admins);
}
mgInfo->bots.remove(user);
if (mgInfo->bots.empty() && mgInfo->botStatus > 0) {
mgInfo->botStatus = -1;
}
}
Data::ChannelAdminChanges(megagroup).remove(uid);
}
}, [&](const MTPDmessageActionChatEditPhoto &data) {
data.vphoto().match([&](const MTPDphoto &data) {
using Flag = MTPDchatPhoto::Flag;
const auto photo = owner().processPhoto(data);
photo->peer = peer;
const auto chatPhoto = MTP_chatPhoto(
MTP_flags((photo->hasVideo() ? Flag::f_has_video : Flag(0))
| (photo->inlineThumbnailBytes().isEmpty()
? Flag(0)
: Flag::f_stripped_thumb)),
MTP_long(photo->id),
MTP_bytes(photo->inlineThumbnailBytes()),
data.vdc_id());
if (const auto chat = peer->asChat()) {
chat->setPhoto(chatPhoto);
} else if (const auto channel = peer->asChannel()) {
channel->setPhoto(chatPhoto);
}
peer->loadUserpic();
}, [&](const MTPDphotoEmpty &data) {
if (const auto chat = peer->asChat()) {
chat->setPhoto(MTP_chatPhotoEmpty());
} else if (const auto channel = peer->asChannel()) {
channel->setPhoto(MTP_chatPhotoEmpty());
}
});
}, [&](const MTPDmessageActionChatEditTitle &data) {
if (const auto chat = peer->asChat()) {
chat->setName(qs(data.vtitle()));
}
}, [&](const MTPDmessageActionChatMigrateTo &data) {
if (const auto chat = peer->asChat()) {
chat->addFlags(ChatDataFlag::Deactivated);
if (const auto channel = owner().channelLoaded(
data.vchannel_id().v)) {
Data::ApplyMigration(chat, channel);
}
}
}, [&](const MTPDmessageActionChannelMigrateFrom &data) {
if (const auto channel = peer->asChannel()) {
channel->addFlags(ChannelDataFlag::Megagroup);
if (const auto chat = owner().chatLoaded(data.vchat_id().v)) {
Data::ApplyMigration(chat, channel);
}
}
}, [&](const MTPDmessageActionPinMessage &data) {
if (replyTo) {
replyTo->match([&](const MTPDmessageReplyHeader &data) {
const auto id = data.vreply_to_msg_id().value_or_empty();
if (id && item) {
session().storage().add(Storage::SharedMediaAddSlice(
peer->id,
MsgId(0),
Storage::SharedMediaType::Pinned,
{ id },
{ id, ServerMaxMsgId }));
setHasPinnedMessages(true);
if (const auto topic = item->topic()) {
session().storage().add(Storage::SharedMediaAddSlice(
peer->id,
topic->rootId(),
Storage::SharedMediaType::Pinned,
{ id },
{ id, ServerMaxMsgId }));
topic->setHasPinnedMessages(true);
}
}
}, [&](const MTPDmessageReplyStoryHeader &data) {
LOG(("API Error: story reply in messageActionPinMessage."));
});
}
}, [&](const MTPDmessageActionGroupCall &data) {
if (const auto channel = peer->asChannel()) {
channel->setGroupCall(data.vcall());
} else if (const auto chat = peer->asChat()) {
chat->setGroupCall(data.vcall());
}
}, [&](const MTPDmessageActionGroupCallScheduled &data) {
if (const auto channel = peer->asChannel()) {
channel->setGroupCall(data.vcall(), data.vschedule_date().v);
} else if (const auto chat = peer->asChat()) {
chat->setGroupCall(data.vcall(), data.vschedule_date().v);
}
}, [&](const MTPDmessageActionPaymentSent &data) {
if (const auto payment = item->Get<HistoryServicePayment>()) {
auto paid = std::optional<Payments::PaidInvoice>();
if (const auto message = payment->msg) {
if (const auto media = message->media()) {
if (const auto invoice = media->invoice()) {
paid = Payments::CheckoutProcess::InvoicePaid(
message);
}
}
} else if (!payment->slug.isEmpty()) {
using Payments::CheckoutProcess;
paid = Payments::CheckoutProcess::InvoicePaid(
&session(),
payment->slug);
}
if (paid) {
// Toast on a current active window.
Ui::Toast::Show({
.text = tr::lng_payments_success(
tr::now,
lt_amount,
Ui::Text::Bold(payment->amount),
lt_title,
Ui::Text::Bold(paid->title),
Ui::Text::WithEntities),
});
}
}
}, [&](const MTPDmessageActionSetChatTheme &data) {
peer->setThemeEmoji(qs(data.vemoticon()));
}, [&](const MTPDmessageActionSetChatWallPaper &data) {
if (item->out() || data.is_for_both()) {
peer->setWallPaper(
Data::WallPaper::Create(&session(), data.vwallpaper()),
!item->out() && data.is_for_both());
}
}, [&](const MTPDmessageActionChatJoinedByRequest &data) {
processJoinedPeer(item->from());
}, [&](const MTPDmessageActionTopicCreate &data) {
if (const auto forum = peer->forum()) {
forum->applyTopicAdded(
item->id,
qs(data.vtitle()),
data.vicon_color().v,
data.vicon_emoji_id().value_or(DocumentId()),
item->from()->id,
item->date(),
item->out());
}
}, [&](const MTPDmessageActionTopicEdit &data) {
if (const auto topic = item->topic()) {
if (const auto &title = data.vtitle()) {
topic->applyTitle(qs(*title));
}
if (const auto icon = data.vicon_emoji_id()) {
topic->applyIconId(icon->v);
}
if (const auto closed = data.vclosed()) {
topic->setClosed(mtpIsTrue(*closed));
}
if (const auto hidden = data.vhidden()) {
topic->setHidden(mtpIsTrue(*hidden));
}
}
}, [](const auto &) {
});
}
void History::mainViewRemoved(
not_null<HistoryBlock*> block,
not_null<HistoryView::Element*> view) {
Expects(_joinedMessage != view->data());
if (_firstUnreadView == view) {
getNextFirstUnreadMessage();
}
if (_unreadBarView == view) {
_unreadBarView = nullptr;
}
if (scrollTopItem == view) {
getNextScrollTopItem(block, view->indexInBlock());
}
}
void History::newItemAdded(not_null<HistoryItem*> item) {
item->indexAsNewItem();
item->addToMessagesIndex();
if (const auto from = item->from() ? item->from()->asUser() : nullptr) {
if (from == item->author()) {
_sendActionPainter.clear(from);
owner().sendActionManager().repliesPaintersClear(this, from);
}
from->madeAction(item->date());
}
item->contributeToSlowmode();
auto notification = Data::ItemNotification{
.item = item,
.type = Data::ItemNotificationType::Message,
};
if (item->showNotification()) {
item->notificationThread()->pushNotification(notification);
}
owner().notifyNewItemAdded(item);
const auto stillShow = item->showNotification(); // Could be read already.
if (stillShow) {
Core::App().notifications().schedule(notification);
}
if (item->out()) {
if (item->isFromScheduled() && unreadCountRefreshNeeded(item->id)) {
if (unreadCountKnown()) {
setUnreadCount(unreadCount() + 1);
} else if (!isForum()) {
owner().histories().requestDialogEntry(this);
}
} else {
destroyUnreadBar();
}
if (!item->unread(this)) {
outboxRead(item);
}
if (item->changesWallPaper()) {
peer->updateFullForced();
}
} else {
if (item->unread(this)) {
if (unreadCountKnown()) {
setUnreadCount(unreadCount() + 1);
} else if (!isForum()) {
owner().histories().requestDialogEntry(this);
}
} else {
inboxRead(item);
}
}
item->incrementReplyToTopCounter();
if (!folderKnown()) {
owner().histories().requestDialogEntry(this);
}
if (const auto topic = item->topic()) {
topic->applyItemAdded(item);
}
}
void History::registerClientSideMessage(not_null<HistoryItem*> item) {
Expects(item->isHistoryEntry());
Expects(IsClientMsgId(item->id));
_clientSideMessages.emplace(item);
session().changes().historyUpdated(this, UpdateFlag::ClientSideMessages);
}
void History::unregisterClientSideMessage(not_null<HistoryItem*> item) {
const auto removed = _clientSideMessages.remove(item);
Assert(removed);
session().changes().historyUpdated(this, UpdateFlag::ClientSideMessages);
}
const base::flat_set<not_null<HistoryItem*>> &History::clientSideMessages() {
return _clientSideMessages;
}
HistoryItem *History::latestSendingMessage() const {
auto sending = ranges::views::all(
_clientSideMessages
) | ranges::views::filter([](not_null<HistoryItem*> item) {
return item->isSending();
});
const auto i = ranges::max_element(sending, ranges::less(), [](
not_null<HistoryItem*> item) {
return std::pair(item->date(), item->id.bare);
});
return (i == sending.end()) ? nullptr : i->get();
}
HistoryBlock *History::prepareBlockForAddingItem() {
if (isBuildingFrontBlock()) {
if (_buildingFrontBlock->block) {
return _buildingFrontBlock->block;
}
blocks.push_front(std::make_unique<HistoryBlock>(this));
for (auto i = 0, l = int(blocks.size()); i != l; ++i) {
blocks[i]->setIndexInHistory(i);
}
_buildingFrontBlock->block = blocks.front().get();
if (_buildingFrontBlock->expectedItemsCount > 0) {
_buildingFrontBlock->block->messages.reserve(
_buildingFrontBlock->expectedItemsCount + 1);
}
return _buildingFrontBlock->block;
}
const auto addNewBlock = blocks.empty()
|| (blocks.back()->messages.size() >= kNewBlockEachMessage);
if (addNewBlock) {
blocks.push_back(std::make_unique<HistoryBlock>(this));
blocks.back()->setIndexInHistory(blocks.size() - 1);
blocks.back()->messages.reserve(kNewBlockEachMessage);
}
return blocks.back().get();
}
void History::viewReplaced(not_null<const Element*> was, Element *now) {
if (scrollTopItem == was) scrollTopItem = now;
if (_firstUnreadView == was) _firstUnreadView = now;
if (_unreadBarView == was) _unreadBarView = now;
}
void History::addItemToBlock(not_null<HistoryItem*> item) {
Expects(!item->mainView());
auto block = prepareBlockForAddingItem();
block->messages.push_back(item->createView(_delegateMixin->delegate()));
const auto view = block->messages.back().get();
view->attachToBlock(block, block->messages.size() - 1);
if (isBuildingFrontBlock() && _buildingFrontBlock->expectedItemsCount > 0) {
--_buildingFrontBlock->expectedItemsCount;
}
}
void History::addEdgesToSharedMedia() {
auto from = loadedAtTop() ? 0 : minMsgId();
auto till = loadedAtBottom() ? ServerMaxMsgId : maxMsgId();
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
const auto type = static_cast<Storage::SharedMediaType>(i);
session().storage().add(Storage::SharedMediaAddSlice(
peer->id,
MsgId(0), // topicRootId
type,
{},
{ from, till }));
}
}
void History::addOlderSlice(const QVector<MTPMessage> &slice) {
if (slice.isEmpty()) {
_loadedAtTop = true;
checkLocalMessages();
return;
}
if (const auto added = createItems(slice); !added.empty()) {
addCreatedOlderSlice(added);
} else {
// If no items were added it means we've loaded everything old.
_loadedAtTop = true;
addEdgesToSharedMedia();
}
checkLocalMessages();
checkLastMessage();
}
void History::addCreatedOlderSlice(
const std::vector<not_null<HistoryItem*>> &items) {
startBuildingFrontBlock(items.size());
for (const auto &item : items) {
addItemToBlock(item);
}
finishBuildingFrontBlock();
if (loadedAtBottom()) {
// Add photos to overview and authors to lastAuthors.
addItemsToLists(items);
for (const auto &item : items) {
if (const auto sublist = item->savedSublist()) {
sublist->applyMaybeLast(item);
}
}
}
addToSharedMedia(items);
}
void History::addNewerSlice(const QVector<MTPMessage> &slice) {
bool wasLoadedAtBottom = loadedAtBottom();
if (slice.isEmpty()) {
_loadedAtBottom = true;
if (!lastMessage()) {
setLastMessage(lastAvailableMessage());
}
}
if (const auto added = createItems(slice); !added.empty()) {
Assert(!isBuildingFrontBlock());
for (const auto &item : added) {
addItemToBlock(item);
}
addToSharedMedia(added);
} else {
_loadedAtBottom = true;
setLastMessage(lastAvailableMessage());
addEdgesToSharedMedia();
}
if (!wasLoadedAtBottom) {
checkAddAllToUnreadMentions();
}
checkLocalMessages();
checkLastMessage();
}
void History::checkLastMessage() {
if (const auto last = lastMessage()) {
if (!_loadedAtBottom && last->mainView()) {
_loadedAtBottom = true;
checkAddAllToUnreadMentions();
}
} else if (_loadedAtBottom) {
setLastMessage(lastAvailableMessage());
}
}
void History::addItemsToLists(
const std::vector<not_null<HistoryItem*>> &items) {
std::deque<not_null<UserData*>> *lastAuthors = nullptr;
base::flat_set<not_null<PeerData*>> *markupSenders = nullptr;
if (peer->isChat()) {
lastAuthors = &peer->asChat()->lastAuthors;
markupSenders = &peer->asChat()->markupSenders;
} else if (peer->isMegagroup()) {
// We don't add users to mgInfo->lastParticipants here.
// We're scrolling back and we see messages from users that
// could be gone from the megagroup already. It is fine for
// chat->lastAuthors, because they're used only for field
// autocomplete, but this is bad for megagroups, because its
// lastParticipants are displayed in Profile as members list.
markupSenders = &peer->asChannel()->mgInfo->markupSenders;
}
for (const auto &item : ranges::views::reverse(items)) {
item->addToUnreadThings(HistoryUnreadThings::AddType::Existing);
if (item->from()->id) {
if (lastAuthors) { // chats
if (auto user = item->from()->asUser()) {
if (!base::contains(*lastAuthors, user)) {
lastAuthors->push_back(user);
}
}
}
}
if (item->author()->id) {
if (markupSenders) { // chats with bots
if (!lastKeyboardInited && item->definesReplyKeyboard() && !item->out()) {
const auto markupFlags = item->replyKeyboardFlags();
if (!(markupFlags & ReplyMarkupFlag::Selective) || item->mentionsMe()) {
bool wasKeyboardHide = markupSenders->contains(item->author());
if (!wasKeyboardHide) {
markupSenders->insert(item->author());
}
if (!(markupFlags & ReplyMarkupFlag::None)) {
if (!lastKeyboardInited) {
bool botNotInChat = false;
if (peer->isChat()) {
botNotInChat = (!Data::CanSendAnything(peer)
|| !peer->asChat()->participants.empty())
&& item->author()->isUser()
&& !peer->asChat()->participants.contains(item->author()->asUser());
} else if (peer->isMegagroup()) {
botNotInChat = (!Data::CanSendAnything(peer)
|| peer->asChannel()->mgInfo->botStatus != 0)
&& item->author()->isUser()
&& !peer->asChannel()->mgInfo->bots.contains(item->author()->asUser());
}
if (wasKeyboardHide || botNotInChat) {
clearLastKeyboard();
} else {
lastKeyboardInited = true;
lastKeyboardId = item->id;
lastKeyboardFrom = item->author()->id;
lastKeyboardUsed = false;
}
}
}
}
}
} else if (!lastKeyboardInited && item->definesReplyKeyboard() && !item->out()) { // conversations with bots
const auto markupFlags = item->replyKeyboardFlags();
if (!(markupFlags & ReplyMarkupFlag::Selective) || item->mentionsMe()) {
if (markupFlags & ReplyMarkupFlag::None) {
clearLastKeyboard();
} else {
lastKeyboardInited = true;
lastKeyboardId = item->id;
lastKeyboardFrom = item->author()->id;
lastKeyboardUsed = false;
}
}
}
}
}
}
void History::checkAddAllToUnreadMentions() {
if (!loadedAtBottom()) {
return;
}
for (const auto &block : blocks) {
for (const auto &message : block->messages) {
const auto item = message->data();
item->addToUnreadThings(HistoryUnreadThings::AddType::Existing);
}
}
}
void History::addToSharedMedia(
const std::vector<not_null<HistoryItem*>> &items) {
std::vector<MsgId> medias[Storage::kSharedMediaTypeCount];
auto topicsWithPinned = base::flat_set<not_null<Data::ForumTopic*>>();
for (const auto &item : items) {
if (const auto types = item->sharedMediaTypes()) {
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
const auto type = static_cast<Storage::SharedMediaType>(i);
if (types.test(type)) {
if (medias[i].empty()) {
medias[i].reserve(items.size());
}
medias[i].push_back(item->id);
if (type == Storage::SharedMediaType::Pinned) {
if (const auto topic = item->topic()) {
if (!topic->hasPinnedMessages()) {
topicsWithPinned.emplace(topic);
}
}
}
}
}
}
}
const auto from = loadedAtTop() ? 0 : minMsgId();
const auto till = loadedAtBottom() ? ServerMaxMsgId : maxMsgId();
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
if (!medias[i].empty()) {
const auto type = static_cast<Storage::SharedMediaType>(i);
session().storage().add(Storage::SharedMediaAddSlice(
peer->id,
MsgId(0), // topicRootId
type,
std::move(medias[i]),
{ from, till }));
if (type == Storage::SharedMediaType::Pinned) {
setHasPinnedMessages(true);
}
}
}
for (const auto &topic : topicsWithPinned) {
topic->setHasPinnedMessages(true);
}
}
void History::calculateFirstUnreadMessage() {
if (!_inboxReadBefore) {
return;
}
_firstUnreadView = nullptr;
if (!unreadCount() || !trackUnreadMessages()) {
return;
}
for (const auto &block : ranges::views::reverse(blocks)) {
for (const auto &message : ranges::views::reverse(block->messages)) {
const auto item = message->data();
if (!item->isRegular()) {
continue;
} else if (!item->out()) {
if (item->id >= *_inboxReadBefore) {
_firstUnreadView = message.get();
} else {
return;
}
}
}
}
}
bool History::readInboxTillNeedsRequest(MsgId tillId) {
Expects(!tillId || IsServerMsgId(tillId));
readClientSideMessages();
if (unreadMark()) {
owner().histories().changeDialogUnreadMark(this, false);
}
DEBUG_LOG(("Reading: readInboxTillNeedsRequest is_server %1, before %2."
).arg(Logs::b(IsServerMsgId(tillId))
).arg(_inboxReadBefore.value_or(-666).bare));
return IsServerMsgId(tillId) && (_inboxReadBefore.value_or(1) <= tillId);
}
void History::readClientSideMessages() {
auto &histories = owner().histories();
for (const auto &item : _clientSideMessages) {
histories.readClientSideMessage(item);
}
}
bool History::unreadCountRefreshNeeded(MsgId readTillId) const {
return !unreadCountKnown()
|| ((readTillId + 1) > _inboxReadBefore.value_or(0));
}
std::optional<int> History::countStillUnreadLocal(MsgId readTillId) const {
if (isEmpty() || !folderKnown()) {
DEBUG_LOG(("Reading: countStillUnreadLocal unknown %1 and %2.").arg(
Logs::b(isEmpty()),
Logs::b(folderKnown())));
return std::nullopt;
}
if (_inboxReadBefore) {
const auto before = *_inboxReadBefore;
DEBUG_LOG(("Reading: check before %1 with min %2 and max %3."
).arg(before.bare
).arg(minMsgId().bare
).arg(maxMsgId().bare));
if (minMsgId() <= before && maxMsgId() >= readTillId) {
auto result = 0;
for (const auto &block : blocks) {
for (const auto &message : block->messages) {
const auto item = message->data();
if (!item->isRegular()
|| (item->out() && !item->isFromScheduled())) {
continue;
} else if (item->id > readTillId) {
break;
} else if (item->id >= before) {
++result;
}
}
}
DEBUG_LOG(("Reading: check before result %1 with existing %2"
).arg(result
).arg(_unreadCount.value_or(-666)));
if (_unreadCount) {
return std::max(*_unreadCount - result, 0);
}
}
}
const auto minimalServerId = minMsgId();
DEBUG_LOG(("Reading: check at end loaded from %1 loaded %2 - %3").arg(
QString::number(minimalServerId.bare),
Logs::b(loadedAtBottom()),
Logs::b(loadedAtTop())));
if (!loadedAtBottom()
|| (!loadedAtTop() && !minimalServerId)
|| minimalServerId > readTillId) {
return std::nullopt;
}
auto result = 0;
for (const auto &block : ranges::views::reverse(blocks)) {
for (const auto &message : ranges::views::reverse(block->messages)) {
const auto item = message->data();
if (item->isRegular()) {
if (item->id <= readTillId) {
return result;
} else if (!item->out()) {
++result;
}
}
}
}
DEBUG_LOG(("Reading: check at end counted %1").arg(result));
return result;
}
void History::applyInboxReadUpdate(
FolderId folderId,
MsgId upTo,
int stillUnread,
int32 channelPts) {
const auto folder = folderId ? owner().folderLoaded(folderId) : nullptr;
if (folder && this->folder() != folder) {
// If history folder is unknown or not synced, request both.
owner().histories().requestDialogEntry(this);
owner().histories().requestDialogEntry(folder);
}
if (_inboxReadBefore.value_or(1) <= upTo) {
if (!peer->isChannel() || peer->asChannel()->pts() == channelPts) {
inboxRead(upTo, stillUnread);
} else {
inboxRead(upTo);
}
}
}
void History::inboxRead(MsgId upTo, std::optional<int> stillUnread) {
if (stillUnread.has_value() && folderKnown()) {
setUnreadCount(*stillUnread);
} else if (const auto still = countStillUnreadLocal(upTo)) {
setUnreadCount(*still);
} else {
owner().histories().requestDialogEntry(this);
}
setInboxReadTill(upTo);
updateChatListEntry();
if (const auto to = peer->migrateTo()) {
if (const auto migrated = peer->owner().historyLoaded(to->id)) {
migrated->updateChatListEntry();
}
}
_firstUnreadView = nullptr;
Core::App().notifications().clearIncomingFromHistory(this);
}
void History::inboxRead(not_null<const HistoryItem*> wasRead) {
if (wasRead->isRegular()) {
inboxRead(wasRead->id);
}
}
void History::outboxRead(MsgId upTo) {
setOutboxReadTill(upTo);
if (const auto last = chatListMessage()) {
if (last->out() && last->isRegular() && last->id <= upTo) {
session().changes().messageUpdated(
last,
Data::MessageUpdate::Flag::DialogRowRepaint);
}
}
updateChatListEntry();
session().changes().historyUpdated(this, UpdateFlag::OutboxRead);
}
void History::outboxRead(not_null<const HistoryItem*> wasRead) {
if (wasRead->isRegular()) {
outboxRead(wasRead->id);
}
}
MsgId History::loadAroundId() const {
if (_unreadCount && *_unreadCount > 0 && _inboxReadBefore) {
return *_inboxReadBefore;
}
return MsgId(0);
}
MsgId History::inboxReadTillId() const {
return _inboxReadBefore.value_or(1) - 1;
}
MsgId History::outboxReadTillId() const {
return _outboxReadBefore.value_or(1) - 1;
}
HistoryItem *History::lastAvailableMessage() const {
return isEmpty() ? nullptr : blocks.back()->messages.back()->data().get();
}
int History::unreadCount() const {
return _unreadCount ? *_unreadCount : 0;
}
bool History::unreadCountKnown() const {
return _unreadCount.has_value();
}
void History::setUnreadCount(int newUnreadCount) {
Expects(folderKnown());
if (_unreadCount == newUnreadCount) {
return;
}
const auto notifier = unreadStateChangeNotifier(!isForum());
_unreadCount = newUnreadCount;
const auto lastOutgoing = [&] {
const auto last = lastMessage();
return last
&& last->isRegular()
&& loadedAtBottom()
&& !isEmpty()
&& blocks.back()->messages.back()->data() == last
&& last->out();
}();
if (newUnreadCount == 1 && !lastOutgoing) {
if (loadedAtBottom()) {
_firstUnreadView = !isEmpty()
? blocks.back()->messages.back().get()
: nullptr;
}
if (const auto last = msgIdForRead()) {
setInboxReadTill(last - 1);
}
} else if (!newUnreadCount) {
_firstUnreadView = nullptr;
if (const auto last = msgIdForRead()) {
setInboxReadTill(last);
}
} else if (!_firstUnreadView && !_unreadBarView && loadedAtBottom()) {
calculateFirstUnreadMessage();
}
}
void History::setUnreadMark(bool unread) {
if (clearUnreadOnClientSide()) {
unread = false;
}
if (unreadMark() == unread) {
return;
}
const auto notifier = unreadStateChangeNotifier(
!unreadCount() && !isForum());
Thread::setUnreadMarkFlag(unread);
}
void History::setFakeUnreadWhileOpened(bool enabled) {
if (fakeUnreadWhileOpened() == enabled) {
return;
} else if (enabled) {
if (!inChatList()) {
return;
}
const auto state = chatListBadgesState();
if (!state.unread && !state.mention) {
return;
}
}
if (enabled) {
_flags |= Flag::FakeUnreadWhileOpened;
} else {
_flags &= ~Flag::FakeUnreadWhileOpened;
}
owner().chatsFilters().refreshHistory(this);
}
[[nodiscard]] bool History::fakeUnreadWhileOpened() const {
return (_flags & Flag::FakeUnreadWhileOpened);
}
void History::setMuted(bool muted) {
if (this->muted() == muted) {
return;
} else {
const auto state = isForum()
? Dialogs::BadgesState()
: computeBadgesState();
const auto notify = (state.unread || state.reaction);
const auto notifier = unreadStateChangeNotifier(notify);
Thread::setMuted(muted);
}
session().changes().peerUpdated(
peer,
Data::PeerUpdate::Flag::Notifications);
owner().chatsFilters().refreshHistory(this);
if (const auto forum = peer->forum()) {
owner().notifySettings().forumParentMuteUpdated(forum);
}
}
void History::getNextFirstUnreadMessage() {
Expects(_firstUnreadView != nullptr);
const auto block = _firstUnreadView->block();
const auto index = _firstUnreadView->indexInBlock();
const auto setFromMessage = [&](const auto &view) {
if (view->data()->isRegular()) {
_firstUnreadView = view.get();
return true;
}
return false;
};
if (index >= 0) {
const auto count = int(block->messages.size());
for (auto i = index + 1; i != count; ++i) {
const auto &message = block->messages[i];
if (setFromMessage(message)) {
return;
}
}
}
const auto count = int(blocks.size());
for (auto j = block->indexInHistory() + 1; j != count; ++j) {
for (const auto &message : blocks[j]->messages) {
if (setFromMessage(message)) {
return;
}
}
}
_firstUnreadView = nullptr;
}
MsgId History::nextNonHistoryEntryId() {
return owner().nextNonHistoryEntryId();
}
bool History::folderKnown() const {
return _folder.has_value();
}
Data::Folder *History::folder() const {
return _folder.value_or(nullptr);
}
void History::setFolder(
not_null<Data::Folder*> folder,
HistoryItem *folderDialogItem) {
setFolderPointer(folder);
if (folderDialogItem) {
setLastServerMessage(folderDialogItem);
}
}
void History::clearFolder() {
setFolderPointer(nullptr);
}
void History::setFolderPointer(Data::Folder *folder) {
if (_folder == folder) {
return;
}
if (isPinnedDialog(FilterId())) {
owner().setChatPinned(this, FilterId(), false);
}
const auto wasKnown = folderKnown();
const auto wasInList = inChatList();
if (wasInList) {
removeFromChatList(0, owner().chatsList(this->folder()));
}
const auto was = _folder.value_or(nullptr);
_folder = folder;
if (was) {
was->unregisterOne(this);
}
if (wasInList) {
addToChatList(0, owner().chatsList(folder));
owner().chatsFilters().refreshHistory(this);
updateChatListEntry();
owner().chatsListChanged(was);
owner().chatsListChanged(folder);
} else if (!wasKnown) {
updateChatListSortPosition();
}
if (folder) {
folder->registerOne(this);
}
session().changes().historyUpdated(this, UpdateFlag::Folder);
}
int History::chatListNameVersion() const {
return peer->nameVersion();
}
void History::hasUnreadMentionChanged(bool has) {
if (isForum()) {
return;
}
auto was = chatListUnreadState();
if (has) {
was.mentions = 0;
} else {
was.mentions = 1;
}
notifyUnreadStateChange(was);
}
void History::hasUnreadReactionChanged(bool has) {
if (isForum()) {
return;
}
auto was = chatListUnreadState();
if (has) {
was.reactions = was.reactionsMuted = 0;
} else {
was.reactions = 1;
was.reactionsMuted = muted() ? was.reactions : 0;
}
notifyUnreadStateChange(was);
}
void History::applyPinnedUpdate(const MTPDupdateDialogPinned &data) {
const auto folderId = data.vfolder_id().value_or_empty();
if (!folderKnown()) {
if (folderId) {
setFolder(owner().folder(folderId));
} else {
clearFolder();
}
}
owner().setChatPinned(this, FilterId(), data.is_pinned());
}
TimeId History::adjustedChatListTimeId() const {
const auto result = chatListTimeId();
if (const auto draft = cloudDraft(MsgId(0))) {
if (!peer->forum()
&& !Data::DraftIsNull(draft)
&& !session().supportMode()) {
return std::max(result, draft->date);
}
}
return result;
}
void History::countScrollState(int top) {
std::tie(scrollTopItem, scrollTopOffset) = findItemAndOffset(top);
}
auto History::findItemAndOffset(int top) const -> std::pair<Element*, int> {
if (const auto element = findScrollTopItem(top)) {
return { element, (top - element->block()->y() - element->y()) };
}
return {};
}
auto History::findScrollTopItem(int top) const -> Element* {
if (isEmpty()) {
return nullptr;
}
auto itemIndex = 0;
auto blockIndex = 0;
auto itemTop = 0;
if (scrollTopItem) {
itemIndex = scrollTopItem->indexInBlock();
blockIndex = scrollTopItem->block()->indexInHistory();
itemTop = blocks[blockIndex]->y() + scrollTopItem->y();
}
if (itemTop > top) {
// go backward through history while we don't find an item that starts above
do {
const auto &block = blocks[blockIndex];
for (--itemIndex; itemIndex >= 0; --itemIndex) {
const auto view = block->messages[itemIndex].get();
itemTop = block->y() + view->y();
if (itemTop <= top) {
return view;
}
}
if (--blockIndex >= 0) {
itemIndex = blocks[blockIndex]->messages.size();
} else {
break;
}
} while (true);
return blocks.front()->messages.front().get();
}
// go forward through history while we don't find the last item that starts above
for (auto blocksCount = int(blocks.size()); blockIndex < blocksCount; ++blockIndex) {
const auto &block = blocks[blockIndex];
for (auto itemsCount = int(block->messages.size()); itemIndex < itemsCount; ++itemIndex) {
itemTop = block->y() + block->messages[itemIndex]->y();
if (itemTop > top) {
Assert(itemIndex > 0 || blockIndex > 0);
if (itemIndex > 0) {
return block->messages[itemIndex - 1].get();
}
return blocks[blockIndex - 1]->messages.back().get();
}
}
itemIndex = 0;
}
return blocks.back()->messages.back().get();
}
void History::getNextScrollTopItem(HistoryBlock *block, int32 i) {
++i;
if (i > 0 && i < block->messages.size()) {
scrollTopItem = block->messages[i].get();
return;
}
int j = block->indexInHistory() + 1;
if (j > 0 && j < blocks.size()) {
scrollTopItem = blocks[j]->messages.front().get();
return;
}
scrollTopItem = nullptr;
}
void History::addUnreadBar() {
if (!_unreadBarView && _firstUnreadView && unreadCount()) {
_unreadBarView = _firstUnreadView;
_unreadBarView->createUnreadBar(tr::lng_unread_bar_some());
}
}
void History::destroyUnreadBar() {
if (const auto view = base::take(_unreadBarView)) {
view->destroyUnreadBar();
}
}
void History::unsetFirstUnreadMessage() {
_firstUnreadView = nullptr;
}
HistoryView::Element *History::unreadBar() const {
return _unreadBarView;
}
HistoryView::Element *History::firstUnreadMessage() const {
return _firstUnreadView;
}
not_null<HistoryItem*> History::addNewInTheMiddle(
not_null<HistoryItem*> item,
int blockIndex,
int itemIndex) {
Expects(blockIndex >= 0);
Expects(blockIndex < blocks.size());
Expects(itemIndex >= 0);
Expects(itemIndex <= blocks[blockIndex]->messages.size());
const auto &block = blocks[blockIndex];
const auto it = block->messages.insert(
block->messages.begin() + itemIndex,
item->createView(_delegateMixin->delegate()));
(*it)->attachToBlock(block.get(), itemIndex);
if (itemIndex + 1 < block->messages.size()) {
for (auto i = itemIndex + 1, l = int(block->messages.size()); i != l; ++i) {
block->messages[i]->setIndexInBlock(i);
}
block->messages[itemIndex + 1]->previousInBlocksChanged();
} else if (blockIndex + 1 < blocks.size() && !blocks[blockIndex + 1]->messages.empty()) {
blocks[blockIndex + 1]->messages.front()->previousInBlocksChanged();
} else {
(*it)->nextInBlocksRemoved();
}
return item;
}
History *History::migrateSibling() const {
const auto addFromId = [&] {
if (const auto from = peer->migrateFrom()) {
return from->id;
} else if (const auto to = peer->migrateTo()) {
return to->id;
}
return PeerId(0);
}();
return owner().historyLoaded(addFromId);
}
Dialogs::UnreadState History::chatListUnreadState() const {
if (const auto forum = peer->forum()) {
return forum->topicsList()->unreadState();
}
return computeUnreadState();
}
Dialogs::BadgesState History::chatListBadgesState() const {
if (const auto forum = peer->forum()) {
return adjustBadgesStateByFolder(
Dialogs::BadgesForUnread(
forum->topicsList()->unreadState(),
Dialogs::CountInBadge::Chats,
Dialogs::IncludeInBadge::UnmutedOrAll));
}
return computeBadgesState();
}
Dialogs::BadgesState History::computeBadgesState() const {
return adjustBadgesStateByFolder(
Dialogs::BadgesForUnread(
computeUnreadState(),
Dialogs::CountInBadge::Messages,
Dialogs::IncludeInBadge::All));
}
Dialogs::BadgesState History::adjustBadgesStateByFolder(
Dialogs::BadgesState state) const {
if (folder()) {
state.mentionMuted = state.reactionMuted = state.unreadMuted = true;
}
return state;
}
Dialogs::UnreadState History::computeUnreadState() const {
auto result = Dialogs::UnreadState();
const auto count = _unreadCount.value_or(0);
const auto mark = !count && unreadMark();
const auto muted = this->muted();
result.messages = count;
result.chats = count ? 1 : 0;
result.marks = mark ? 1 : 0;
result.mentions = unreadMentions().has() ? 1 : 0;
result.reactions = unreadReactions().has() ? 1 : 0;
result.messagesMuted = muted ? result.messages : 0;
result.chatsMuted = muted ? result.chats : 0;
result.marksMuted = muted ? result.marks : 0;
result.reactionsMuted = muted ? result.reactions : 0;
result.known = _unreadCount.has_value();
return result;
}
void History::allowChatListMessageResolve() {
if (_flags & Flag::ResolveChatListMessage) {
return;
}
_flags |= Flag::ResolveChatListMessage;
if (!chatListMessageKnown()) {
requestChatListMessage();
} else {
resolveChatListMessageGroup();
}
}
void History::resolveChatListMessageGroup() {
const auto item = _chatListMessage.value_or(nullptr);
if (!(_flags & Flag::ResolveChatListMessage)
|| !item
|| !hasOrphanMediaGroupPart()) {
return;
}
// If we set a single album part, request the full album.
const auto withImages = !item->toPreview({
.hideSender = true,
.hideCaption = true }).images.empty();
if (withImages) {
owner().histories().requestGroupAround(item);
}
if (unreadCountKnown() && !unreadCount()) {
// When we add just one last item, like we do while loading dialogs,
// we want to remove a single added grouped media, otherwise it will
// jump once we open the message history (first we show only that
// media, then we load the rest of the group and show the group).
//
// That way when we open the message history we show nothing until a
// whole history part is loaded, it certainly will contain the group.
clear(ClearType::Unload);
}
}
HistoryItem *History::chatListMessage() const {
return _chatListMessage.value_or(nullptr);
}
bool History::chatListMessageKnown() const {
return _chatListMessage.has_value();
}
const QString &History::chatListName() const {
return peer->name();
}
const QString &History::chatListNameSortKey() const {
return _chatListNameSortKey;
}
void History::refreshChatListNameSortKey() {
_chatListNameSortKey = owner().nameSortKey(peer->name());
}
const base::flat_set<QString> &History::chatListNameWords() const {
return peer->nameWords();
}
const base::flat_set<QChar> &History::chatListFirstLetters() const {
return peer->nameFirstLetters();
}
void History::chatListPreloadData() {
peer->loadUserpic();
allowChatListMessageResolve();
}
void History::paintUserpic(
Painter &p,
Ui::PeerUserpicView &view,
const Dialogs::Ui::PaintContext &context) const {
peer->paintUserpic(
p,
view,
context.st->padding.left(),
context.st->padding.top(),
context.st->photoSize);
}
void History::startBuildingFrontBlock(int expectedItemsCount) {
Assert(!isBuildingFrontBlock());
Assert(expectedItemsCount > 0);
_buildingFrontBlock = std::make_unique<BuildingBlock>();
_buildingFrontBlock->expectedItemsCount = expectedItemsCount;
}
void History::finishBuildingFrontBlock() {
Expects(isBuildingFrontBlock());
// Some checks if there was some message history already
if (const auto block = base::take(_buildingFrontBlock)->block) {
if (blocks.size() > 1) {
// ... item, item, item, last ], [ first, item, item ...
const auto first = blocks[1]->messages.front().get();
// we've added a new front block, so previous item for
// the old first item of a first block was changed
first->previousInBlocksChanged();
} else {
block->messages.back()->nextInBlocksRemoved();
}
}
}
bool History::loadedAtBottom() const {
return _loadedAtBottom;
}
bool History::loadedAtTop() const {
return _loadedAtTop;
}
bool History::isReadyFor(MsgId msgId) {
if (msgId < 0 && -msgId < ServerMaxMsgId && peer->migrateFrom()) {
// Old group history.
return owner().history(peer->migrateFrom()->id)->isReadyFor(-msgId);
}
if (msgId == ShowAtTheEndMsgId) {
return loadedAtBottom();
}
if (msgId == ShowAtUnreadMsgId) {
if (const auto migratePeer = peer->migrateFrom()) {
if (const auto migrated = owner().historyLoaded(migratePeer)) {
if (migrated->unreadCount()) {
return migrated->isReadyFor(msgId);
}
}
}
if (unreadCount() && _inboxReadBefore) {
if (!isEmpty()) {
return (loadedAtTop() || minMsgId() <= *_inboxReadBefore)
&& (loadedAtBottom() || maxMsgId() >= *_inboxReadBefore);
}
return false;
}
return loadedAtBottom();
}
const auto item = owner().message(peer, msgId);
return item && (item->history() == this) && item->mainView();
}
void History::getReadyFor(MsgId msgId) {
if (msgId < 0 && -msgId < ServerMaxMsgId && peer->migrateFrom()) {
const auto migrated = owner().history(peer->migrateFrom()->id);
migrated->getReadyFor(-msgId);
if (migrated->isEmpty()) {
clear(ClearType::Unload);
}
return;
}
if (msgId == ShowAtUnreadMsgId) {
if (const auto migratePeer = peer->migrateFrom()) {
if (const auto migrated = owner().historyLoaded(migratePeer)) {
if (migrated->unreadCount()) {
clear(ClearType::Unload);
migrated->getReadyFor(msgId);
return;
}
}
}
}
if (!isReadyFor(msgId)) {
clear(ClearType::Unload);
if (const auto migratePeer = peer->migrateFrom()) {
if (const auto migrated = owner().historyLoaded(migratePeer)) {
migrated->clear(ClearType::Unload);
}
}
if ((msgId == ShowAtTheEndMsgId)
|| (msgId == ShowAtUnreadMsgId && !unreadCount())) {
_loadedAtBottom = true;
}
}
}
void History::setNotLoadedAtBottom() {
_loadedAtBottom = false;
session().storage().invalidate(
Storage::SharedMediaInvalidateBottom(peer->id));
if (const auto messages = _messages.get()) {
messages->invalidateBottom();
}
}
void History::clearSharedMedia() {
session().storage().remove(
Storage::SharedMediaRemoveAll(peer->id));
}
void History::setLastServerMessage(HistoryItem *item) {
_lastServerMessage = item;
if (_lastMessage
&& *_lastMessage
&& !(*_lastMessage)->isRegular()
&& (!item || (*_lastMessage)->date() > item->date())) {
return;
}
setLastMessage(item);
}
void History::setLastMessage(HistoryItem *item) {
if (_lastMessage && *_lastMessage == item) {
return;
}
_lastMessage = item;
if (!item || item->isRegular()) {
_lastServerMessage = item;
}
if (peer->migrateTo()) {
// We don't want to request last message for all deactivated chats.
// This is a heavy request for them, because we need to get last
// two items by messages.getHistory to skip the migration message.
setChatListMessageUnknown();
} else {
setChatListMessageFromLast();
if (!chatListMessageKnown()) {
setFakeChatListMessage();
}
}
}
void History::refreshChatListMessage() {
const auto known = chatListMessageKnown();
setChatListMessageFromLast();
if (known && !_chatListMessage) {
requestChatListMessage();
}
}
void History::setChatListMessage(HistoryItem *item) {
if (_chatListMessage && *_chatListMessage == item) {
return;
}
const auto was = _chatListMessage.value_or(nullptr);
if (item) {
if (item->isSponsored()) {
return;
}
if (_chatListMessage
&& *_chatListMessage
&& !(*_chatListMessage)->isRegular()
&& (*_chatListMessage)->date() > item->date()) {
return;
}
_chatListMessage = item;
setChatListTimeId(item->date());
resolveChatListMessageGroup();
} else if (!_chatListMessage || *_chatListMessage) {
_chatListMessage = nullptr;
updateChatListEntry();
}
if (const auto folder = this->folder()) {
folder->oneListMessageChanged(was, item);
}
if (const auto to = peer->migrateTo()) {
if (const auto history = owner().historyLoaded(to)) {
if (!history->chatListMessageKnown()) {
history->requestChatListMessage();
}
}
}
}
auto History::computeChatListMessageFromLast() const
-> std::optional<HistoryItem*> {
if (!_lastMessage) {
return _lastMessage;
}
// In migrated groups we want to skip essential message
// about migration in the chats list and display the last
// non-migration message from the original legacy group.
const auto last = lastMessage();
if (!last || !last->isGroupMigrate()) {
return _lastMessage;
}
if (const auto chat = peer->asChat()) {
// In chats we try to take the item before the 'last', which
// is the empty-displayed migration message.
if (!loadedAtBottom()) {
// We don't know the tail of the history.
return std::nullopt;
}
const auto before = [&]() -> HistoryItem* {
for (const auto &block : ranges::views::reverse(blocks)) {
const auto &messages = block->messages;
for (const auto &item : ranges::views::reverse(messages)) {
if (item->data() != last) {
return item->data();
}
}
}
return nullptr;
}();
if (before) {
// We found a message that is not the migration one.
return before;
} else if (loadedAtTop()) {
// No other messages in this history.
return _lastMessage;
}
return std::nullopt;
} else if (const auto from = migrateFrom()) {
// In megagroups we just try to use
// the message from the original group.
return from->chatListMessageKnown()
? std::make_optional(from->chatListMessage())
: std::nullopt;
}
return _lastMessage;
}
void History::setChatListMessageFromLast() {
if (const auto good = computeChatListMessageFromLast()) {
setChatListMessage(*good);
} else {
setChatListMessageUnknown();
}
}
void History::setChatListMessageUnknown() {
if (!_chatListMessage.has_value()) {
return;
}
const auto was = *_chatListMessage;
_chatListMessage = std::nullopt;
if (const auto folder = this->folder()) {
folder->oneListMessageChanged(was, nullptr);
}
}
void History::requestChatListMessage() {
if (!lastMessageKnown()) {
owner().histories().requestDialogEntry(this, [=] {
requestChatListMessage();
});
return;
} else if (chatListMessageKnown()) {
return;
}
setChatListMessageFromLast();
if (!chatListMessageKnown()) {
setFakeChatListMessage();
}
}
void History::setFakeChatListMessage() {
if (!(_flags & Flag::ResolveChatListMessage)) {
if (!chatListTimeId()) {
if (const auto last = lastMessage()) {
setChatListTimeId(last->date());
}
}
return;
} else if (const auto chat = peer->asChat()) {
// In chats we try to take the item before the 'last', which
// is the empty-displayed migration message.
owner().histories().requestFakeChatListMessage(this);
} else if (const auto from = migrateFrom()) {
// In megagroups we just try to use
// the message from the original group.
from->allowChatListMessageResolve();
from->requestChatListMessage();
}
}
void History::setFakeChatListMessageFrom(const MTPmessages_Messages &data) {
if (!lastMessageKnown()) {
requestChatListMessage();
return;
}
const auto finalize = gsl::finally([&] {
// Make sure that we have chatListMessage when we get out of here.
if (!chatListMessageKnown()) {
setChatListMessage(lastMessage());
}
});
const auto last = lastMessage();
if (!last || !last->isGroupMigrate()) {
// Last message is good enough.
return;
}
const auto other = data.match([&](
const MTPDmessages_messagesNotModified &) {
return static_cast<const MTPMessage*>(nullptr);
}, [&](const auto &data) {
for (const auto &message : data.vmessages().v) {
const auto id = message.match([](const auto &data) {
return data.vid().v;
});
if (id != last->id) {
return &message;
}
}
return static_cast<const MTPMessage*>(nullptr);
});
if (!other) {
// Other (non equal to the last one) message not found.
return;
}
const auto item = owner().addNewMessage(
*other,
MessageFlags(),
NewMessageType::Existing);
if (!item || item->isGroupMigrate()) {
// Not better than the last one.
return;
}
setChatListMessage(item);
}
void History::applyChatListGroup(
PeerId dataPeerId,
const MTPmessages_Messages &data) {
if (!isEmpty()
|| !_chatListMessage
|| !*_chatListMessage
|| (*_chatListMessage)->history() != this
|| !_lastMessage
|| !*_lastMessage
|| dataPeerId != peer->id) {
return;
}
// Apply loaded album as a last slice.
const auto processMessages = [&](const MTPVector<MTPMessage> &messages) {
auto items = std::vector<not_null<HistoryItem*>>();
items.reserve(messages.v.size());
for (const auto &message : messages.v) {
const auto id = IdFromMessage(message);
if (const auto message = owner().message(dataPeerId, id)) {
items.push_back(message);
}
}
if (!ranges::contains(items, not_null(*_lastMessage))
|| !ranges::contains(items, not_null(*_chatListMessage))) {
return;
}
_loadedAtBottom = true;
ranges::sort(items, ranges::less{}, &HistoryItem::id);
addCreatedOlderSlice(items);
checkLocalMessages();
checkLastMessage();
};
data.match([&](const MTPDmessages_messagesNotModified &) {
}, [&](const auto &data) {
processMessages(data.vmessages());
});
}
HistoryItem *History::lastMessage() const {
return _lastMessage.value_or(nullptr);
}
bool History::lastMessageKnown() const {
return _lastMessage.has_value();
}
HistoryItem *History::lastServerMessage() const {
return _lastServerMessage.value_or(nullptr);
}
bool History::lastServerMessageKnown() const {
return _lastServerMessage.has_value();
}
void History::updateChatListExistence() {
Entry::updateChatListExistence();
}
bool History::useTopPromotion() const {
if (!isTopPromoted()) {
return false;
} else if (const auto channel = peer->asChannel()) {
return !isPinnedDialog(FilterId()) && !channel->amIn();
} else if (const auto user = peer->asUser()) {
return !isPinnedDialog(FilterId()) && user->isBot() && isEmpty();
}
return false;
}
int History::fixedOnTopIndex() const {
return useTopPromotion() ? kTopPromotionFixOnTopIndex : 0;
}
bool History::trackUnreadMessages() const {
if (const auto channel = peer->asChannel()) {
return channel->amIn();
}
return true;
}
bool History::shouldBeInChatList() const {
if (peer->migrateTo() || !folderKnown()) {
return false;
} else if (isPinnedDialog(FilterId())) {
return true;
} else if (const auto channel = peer->asChannel()) {
if (!channel->amIn()) {
return isTopPromoted();
}
} else if (const auto chat = peer->asChat()) {
return chat->amIn()
|| !lastMessageKnown()
|| (lastMessage() != nullptr);
} else if (const auto user = peer->asUser()) {
if (user->isBot() && isTopPromoted()) {
return true;
}
}
return !lastMessageKnown()
|| (lastMessage() != nullptr);
}
void History::unknownMessageDeleted(MsgId messageId) {
if (_inboxReadBefore && messageId >= *_inboxReadBefore) {
owner().histories().requestDialogEntry(this);
}
}
bool History::isServerSideUnread(not_null<const HistoryItem*> item) const {
Expects(item->isRegular());
return item->out()
? (!_outboxReadBefore || (item->id >= *_outboxReadBefore))
: (!_inboxReadBefore || (item->id >= *_inboxReadBefore));
}
void History::applyDialog(
Data::Folder *requestFolder,
const MTPDdialog &data) {
const auto folderId = data.vfolder_id();
const auto folder = !folderId
? requestFolder
: folderId->v
? owner().folder(folderId->v).get()
: nullptr;
applyDialogFields(
folder,
data.vunread_count().v,
data.vread_inbox_max_id().v,
data.vread_outbox_max_id().v);
applyDialogTopMessage(data.vtop_message().v);
setUnreadMark(data.is_unread_mark());
unreadMentions().setCount(data.vunread_mentions_count().v);
unreadReactions().setCount(data.vunread_reactions_count().v);
if (const auto channel = peer->asChannel()) {
if (const auto pts = data.vpts()) {
channel->ptsReceived(pts->v);
}
if (!channel->amCreator()) {
const auto topMessageId = FullMsgId(
channel->id,
data.vtop_message().v);
if (const auto item = owner().message(topMessageId)) {
if (item->date() <= channel->date) {
session().api().chatParticipants().requestSelf(channel);
}
}
}
channel->setViewAsMessagesFlag(data.is_view_forum_as_messages());
}
owner().notifySettings().apply(
MTP_notifyPeer(data.vpeer()),
data.vnotify_settings());
const auto draft = data.vdraft();
if (draft && draft->type() == mtpc_draftMessage) {
Data::ApplyPeerCloudDraft(
&session(),
peer->id,
MsgId(0), // topicRootId
draft->c_draftMessage());
}
if (const auto ttl = data.vttl_period()) {
peer->setMessagesTTL(ttl->v);
}
owner().histories().dialogEntryApplied(this);
}
void History::dialogEntryApplied() {
if (!lastServerMessageKnown()) {
setLastServerMessage(nullptr);
} else if (!lastMessageKnown()) {
setLastMessage(nullptr);
}
if (peer->migrateTo()) {
return;
} else if (!chatListMessageKnown()) {
requestChatListMessage();
return;
}
if (!chatListMessage()) {
clear(ClearType::Unload);
addNewerSlice(QVector<MTPMessage>());
addOlderSlice(QVector<MTPMessage>());
if (const auto channel = peer->asChannel()) {
const auto inviter = channel->inviter;
if (inviter && channel->amIn()) {
if (const auto from = owner().userLoaded(inviter)) {
insertJoinedMessage();
}
}
}
return;
}
if (chatListTimeId() != 0 && loadedAtBottom()) {
if (const auto channel = peer->asChannel()) {
const auto inviter = channel->inviter;
if (inviter
&& chatListTimeId() <= channel->inviteDate
&& channel->amIn()) {
if (const auto from = owner().userLoaded(inviter)) {
insertJoinedMessage();
}
}
}
}
}
void History::cacheTopPromotion(
bool promoted,
const QString &type,
const QString &message) {
const auto changed = (isTopPromoted() != promoted);
cacheTopPromoted(promoted);
if (topPromotionType() != type || _topPromotedMessage != message) {
_topPromotedType = type;
_topPromotedMessage = message;
cloudDraftTextCache().clear();
} else if (changed) {
cloudDraftTextCache().clear();
}
}
QStringView History::topPromotionType() const {
return topPromotionAboutShown()
? base::StringViewMid(_topPromotedType, 5)
: QStringView(_topPromotedType);
}
bool History::topPromotionAboutShown() const {
return _topPromotedType.startsWith("seen^");
}
void History::markTopPromotionAboutShown() {
if (!topPromotionAboutShown()) {
_topPromotedType = "seen^" + _topPromotedType;
}
}
QString History::topPromotionMessage() const {
return _topPromotedMessage;
}
bool History::clearUnreadOnClientSide() const {
if (!session().supportMode()) {
return false;
}
if (const auto user = peer->asUser()) {
if (user->isInaccessible()) {
return true;
}
}
return false;
}
bool History::skipUnreadUpdate() const {
return clearUnreadOnClientSide();
}
void History::applyDialogFields(
Data::Folder *folder,
int unreadCount,
MsgId maxInboxRead,
MsgId maxOutboxRead) {
if (folder) {
setFolder(folder);
} else {
clearFolder();
}
if (!skipUnreadUpdate()
&& maxInboxRead + 1 >= _inboxReadBefore.value_or(1)) {
setUnreadCount(unreadCount);
setInboxReadTill(maxInboxRead);
}
setOutboxReadTill(maxOutboxRead);
}
void History::applyDialogTopMessage(MsgId topMessageId) {
if (topMessageId) {
const auto itemId = FullMsgId(peer->id, topMessageId);
if (const auto item = owner().message(itemId)) {
setLastServerMessage(item);
} else {
setLastServerMessage(nullptr);
}
} else {
setLastServerMessage(nullptr);
}
if (clearUnreadOnClientSide()) {
setUnreadCount(0);
if (const auto last = lastMessage()) {
setInboxReadTill(last->id);
}
}
}
void History::setInboxReadTill(MsgId upTo) {
if (_inboxReadBefore) {
accumulate_max(*_inboxReadBefore, upTo + 1);
} else {
_inboxReadBefore = upTo + 1;
}
}
void History::setOutboxReadTill(MsgId upTo) {
if (_outboxReadBefore) {
accumulate_max(*_outboxReadBefore, upTo + 1);
} else {
_outboxReadBefore = upTo + 1;
}
}
MsgId History::minMsgId() const {
for (const auto &block : blocks) {
for (const auto &message : block->messages) {
const auto item = message->data();
if (item->isRegular()) {
return item->id;
}
}
}
return 0;
}
MsgId History::maxMsgId() const {
for (const auto &block : ranges::views::reverse(blocks)) {
for (const auto &message : ranges::views::reverse(block->messages)) {
const auto item = message->data();
if (item->isRegular()) {
return item->id;
}
}
}
return 0;
}
MsgId History::msgIdForRead() const {
const auto last = lastMessage();
const auto result = (last && last->isRegular())
? last->id
: MsgId(0);
return loadedAtBottom()
? std::max(result, maxMsgId())
: result;
}
HistoryItem *History::lastEditableMessage() const {
if (!loadedAtBottom()) {
return nullptr;
}
const auto now = base::unixtime::now();
for (const auto &block : ranges::views::reverse(blocks)) {
for (const auto &message : ranges::views::reverse(block->messages)) {
const auto item = message->data();
if (item->allowsEdit(now)) {
return owner().groups().findItemToEdit(item);
}
}
}
return nullptr;
}
void History::resizeToWidth(int newWidth) {
using Request = HistoryBlock::ResizeRequest;
const auto request = (_flags & Flag::PendingAllItemsResize)
? Request::ReinitAll
: (_width != newWidth)
? Request::ResizeAll
: Request::ResizePending;
if (request == Request::ResizePending && !hasPendingResizedItems()) {
return;
}
_flags &= ~(Flag::HasPendingResizedItems | Flag::PendingAllItemsResize);
_width = newWidth;
int y = 0;
for (const auto &block : blocks) {
block->setY(y);
y += block->resizeGetHeight(newWidth, request);
}
_height = y;
}
void History::forceFullResize() {
_width = 0;
_flags |= Flag::HasPendingResizedItems;
}
Data::Thread *History::threadFor(MsgId topicRootId) {
return topicRootId
? peer->forumTopicFor(topicRootId)
: static_cast<Data::Thread*>(this);
}
const Data::Thread *History::threadFor(MsgId topicRootId) const {
return const_cast<History*>(this)->threadFor(topicRootId);
}
void History::forumChanged(Data::Forum *old) {
if (inChatList()) {
notifyUnreadStateChange(old
? old->topicsList()->unreadState()
: computeUnreadState());
}
if (const auto forum = peer->forum()) {
_flags |= Flag::IsForum;
forum->topicsList()->unreadStateChanges(
) | rpl::filter([=] {
return (_flags & Flag::IsForum) && inChatList();
}) | rpl::start_with_next([=](const Dialogs::UnreadState &old) {
notifyUnreadStateChange(old);
}, forum->lifetime());
forum->chatsListChanges(
) | rpl::start_with_next([=] {
updateChatListEntry();
}, forum->lifetime());
} else {
_flags &= ~Flag::IsForum;
}
if (cloudDraft(MsgId(0))) {
updateChatListSortPosition();
}
_flags |= Flag::PendingAllItemsResize;
}
bool History::isForum() const {
return (_flags & Flag::IsForum);
}
not_null<History*> History::migrateToOrMe() const {
if (const auto to = peer->migrateTo()) {
return owner().history(to);
}
// We could get it by owner().history(peer), but we optimize.
return const_cast<History*>(this);
}
History *History::migrateFrom() const {
if (const auto from = peer->migrateFrom()) {
return owner().history(from);
}
return nullptr;
}
MsgRange History::rangeForDifferenceRequest() const {
auto fromId = MsgId(0);
auto toId = MsgId(0);
for (const auto &block : blocks) {
for (const auto &item : block->messages) {
const auto id = item->data()->id;
if (id > 0) {
fromId = id;
break;
}
}
if (fromId) break;
}
if (fromId) {
for (auto blockIndex = blocks.size(); blockIndex > 0;) {
const auto &block = blocks[--blockIndex];
for (auto itemIndex = block->messages.size(); itemIndex > 0;) {
const auto id = block->messages[--itemIndex]->data()->id;
if (id > 0) {
toId = id;
break;
}
}
if (toId) break;
}
return { fromId, toId + 1 };
}
return MsgRange();
}
Data::HistoryMessages &History::messages() {
if (!_messages) {
_messages = std::make_unique<Data::HistoryMessages>();
const auto max = maxMsgId();
const auto from = loadedAtTop() ? 0 : minMsgId();
const auto till = loadedAtBottom() ? ServerMaxMsgId : max;
auto list = std::vector<MsgId>();
list.reserve(std::min(
int(_items.size()),
int(blocks.size()) * kNewBlockEachMessage));
auto sort = false;
for (const auto &block : blocks) {
for (const auto &view : block->messages) {
const auto item = view->data();
if (item->isRegular()) {
const auto id = item->id;
if (!list.empty() && list.back() >= id) {
sort = true;
}
list.push_back(id);
}
}
}
if (sort) {
ranges::sort(list);
}
if (max || (loadedAtTop() && loadedAtBottom())) {
_messages->addSlice(std::move(list), { from, till }, {});
}
}
return *_messages;
}
const Data::HistoryMessages &History::messages() const {
return const_cast<History*>(this)->messages();
}
Data::HistoryMessages *History::maybeMessages() {
return _messages.get();
}
HistoryItem *History::insertJoinedMessage() {
const auto channel = peer->asChannel();
if (!channel
|| _joinedMessage
|| !channel->amIn()
|| (peer->isMegagroup()
&& channel->mgInfo->joinedMessageFound)) {
return _joinedMessage;
}
const auto inviter = (channel->inviter.bare > 0)
? owner().userLoaded(channel->inviter)
: nullptr;
if (!inviter) {
return nullptr;
}
if (peer->isMegagroup()
&& peer->migrateFrom()
&& !blocks.empty()
&& blocks.front()->messages.front()->data()->id == 1) {
channel->mgInfo->joinedMessageFound = true;
return nullptr;
}
_joinedMessage = GenerateJoinedMessage(
this,
channel->inviteDate,
inviter,
channel->inviteViaRequest);
insertMessageToBlocks(_joinedMessage);
return _joinedMessage;
}
void History::insertMessageToBlocks(not_null<HistoryItem*> item) {
Expects(item->mainView() == nullptr);
if (isEmpty()) {
addNewToBack(item, false);
return;
}
const auto itemDate = item->date();
for (auto blockIndex = blocks.size(); blockIndex > 0;) {
const auto &block = blocks[--blockIndex];
for (auto itemIndex = block->messages.size(); itemIndex > 0;) {
if (block->messages[--itemIndex]->data()->date() <= itemDate) {
++itemIndex;
addNewInTheMiddle(item, blockIndex, itemIndex);
const auto lastDate = chatListTimeId();
if (!lastDate || itemDate >= lastDate) {
setLastMessage(item);
owner().notifyHistoryChangeDelayed(this);
}
return;
}
}
}
startBuildingFrontBlock();
addItemToBlock(item);
finishBuildingFrontBlock();
}
void History::checkLocalMessages() {
if (isEmpty() && (!loadedAtTop() || !loadedAtBottom())) {
return;
}
const auto firstDate = loadedAtTop()
? 0
: blocks.front()->messages.front()->data()->date();
const auto lastDate = loadedAtBottom()
? std::numeric_limits<TimeId>::max()
: blocks.back()->messages.back()->data()->date();
const auto goodDate = [&](TimeId date) {
return (date >= firstDate && date < lastDate);
};
for (const auto &item : _clientSideMessages) {
if (!item->mainView() && goodDate(item->date())) {
insertMessageToBlocks(item);
}
}
if (peer->isChannel()
&& !_joinedMessage
&& peer->asChannel()->inviter
&& goodDate(peer->asChannel()->inviteDate)) {
insertJoinedMessage();
}
}
HistoryItem *History::joinedMessageInstance() const {
return _joinedMessage;
}
void History::removeJoinedMessage() {
if (_joinedMessage) {
_joinedMessage->destroy();
}
}
void History::reactionsEnabledChanged(bool enabled) {
if (!enabled) {
for (const auto &item : _items) {
item->updateReactions(nullptr);
}
} else {
for (const auto &item : _items) {
item->updateReactionsUnknown();
}
}
}
bool History::isEmpty() const {
return blocks.empty();
}
bool History::isDisplayedEmpty() const {
if (!loadedAtTop() || !loadedAtBottom()) {
return false;
}
const auto first = findFirstNonEmpty();
if (!first) {
return true;
}
const auto chat = peer->asChat();
if (!chat || !chat->amCreator()) {
return false;
}
// For legacy chats we want to show the chat with only
// messages about you creating the group and maybe about you
// changing the group photo as an empty chat with
// a nice information about the group features.
if (nonEmptyCountMoreThan(2)) {
return false;
}
const auto isChangePhoto = [](not_null<HistoryItem*> item) {
if (const auto media = item->media()) {
return (media->photo() != nullptr) && item->isService();
}
return false;
};
const auto last = findLastNonEmpty();
if (first == last) {
return first->data()->isGroupEssential()
|| isChangePhoto(first->data());
}
return first->data()->isGroupEssential() && isChangePhoto(last->data());
}
auto History::findFirstNonEmpty() const -> Element* {
for (const auto &block : blocks) {
for (const auto &element : block->messages) {
if (!element->data()->isEmpty()) {
return element.get();
}
}
}
return nullptr;
}
auto History::findFirstDisplayed() const -> Element* {
for (const auto &block : blocks) {
for (const auto &element : block->messages) {
if (!element->data()->isEmpty() && !element->isHidden()) {
return element.get();
}
}
}
return nullptr;
}
auto History::findLastNonEmpty() const -> Element* {
for (const auto &block : ranges::views::reverse(blocks)) {
for (const auto &element : ranges::views::reverse(block->messages)) {
if (!element->data()->isEmpty()) {
return element.get();
}
}
}
return nullptr;
}
auto History::findLastDisplayed() const -> Element* {
for (const auto &block : ranges::views::reverse(blocks)) {
for (const auto &element : ranges::views::reverse(block->messages)) {
if (!element->data()->isEmpty() && !element->isHidden()) {
return element.get();
}
}
}
return nullptr;
}
bool History::nonEmptyCountMoreThan(int count) const {
Expects(count >= 0);
for (const auto &block : blocks) {
for (const auto &element : block->messages) {
if (!element->data()->isEmpty()) {
if (!count--) {
return true;
}
}
}
}
return false;
}
bool History::hasOrphanMediaGroupPart() const {
if (loadedAtTop() || !loadedAtBottom()) {
return false;
} else if (blocks.size() != 1) {
return false;
} else if (blocks.front()->messages.size() != 1) {
return false;
}
const auto last = blocks.front()->messages.front()->data();
return last->groupId() != MessageGroupId();
}
std::vector<MsgId> History::collectMessagesFromParticipantToDelete(
not_null<PeerData*> participant) const {
auto result = std::vector<MsgId>();
for (const auto &block : blocks) {
for (const auto &message : block->messages) {
const auto item = message->data();
if (item->from() == participant && item->canDelete()) {
result.push_back(item->id);
}
}
}
return result;
}
void History::clear(ClearType type) {
_unreadBarView = nullptr;
_firstUnreadView = nullptr;
removeJoinedMessage();
forgetScrollState();
blocks.clear();
owner().notifyHistoryUnloaded(this);
lastKeyboardInited = false;
if (type == ClearType::Unload) {
_loadedAtTop = _loadedAtBottom = false;
} else {
// Leave the 'sending' messages in local messages.
auto local = base::flat_set<not_null<HistoryItem*>>();
for (const auto &item : _clientSideMessages) {
if (!item->isSending()) {
local.emplace(item);
}
}
for (const auto &item : local) {
item->destroy();
}
clearNotifications();
owner().notifyHistoryCleared(this);
if (unreadCountKnown()) {
setUnreadCount(0);
}
if (type == ClearType::DeleteChat) {
setLastServerMessage(nullptr);
} else if (_lastMessage && *_lastMessage) {
if ((*_lastMessage)->isRegular()) {
(*_lastMessage)->applyEditionToHistoryCleared();
} else {
_lastMessage = std::nullopt;
}
}
const auto tillId = (_lastMessage && *_lastMessage)
? (*_lastMessage)->id
: std::numeric_limits<MsgId>::max();
clearUpTill(tillId);
if (blocks.empty() && _lastMessage && *_lastMessage) {
addItemToBlock(*_lastMessage);
}
_loadedAtTop = _loadedAtBottom = _lastMessage.has_value();
clearSharedMedia();
if (const auto messages = _messages.get()) {
messages->removeAll();
}
clearLastKeyboard();
}
if (const auto chat = peer->asChat()) {
chat->lastAuthors.clear();
chat->markupSenders.clear();
} else if (const auto channel = peer->asMegagroup()) {
channel->mgInfo->markupSenders.clear();
}
owner().notifyHistoryChangeDelayed(this);
owner().sendHistoryChangeNotifications();
}
void History::clearUpTill(MsgId availableMinId) {
auto remove = std::vector<not_null<HistoryItem*>>();
remove.reserve(_items.size());
for (const auto &item : _items) {
const auto itemId = item->id;
if (!item->isRegular()) {
continue;
} else if (itemId == availableMinId) {
item->applyEditionToHistoryCleared();
} else if (itemId < availableMinId) {
remove.push_back(item.get());
}
}
for (const auto item : remove) {
item->destroy();
}
requestChatListMessage();
}
void History::applyGroupAdminChanges(const base::flat_set<UserId> &changes) {
for (const auto &block : blocks) {
for (const auto &message : block->messages) {
message->applyGroupAdminChanges(changes);
}
}
}
void History::changedChatListPinHook() {
session().changes().historyUpdated(this, UpdateFlag::IsPinned);
}
void History::removeBlock(not_null<HistoryBlock*> block) {
Expects(block->messages.empty());
if (_buildingFrontBlock && block == _buildingFrontBlock->block) {
_buildingFrontBlock->block = nullptr;
}
int index = block->indexInHistory();
blocks.erase(blocks.begin() + index);
if (index < blocks.size()) {
for (int i = index, l = blocks.size(); i < l; ++i) {
blocks[i]->setIndexInHistory(i);
}
blocks[index]->messages.front()->previousInBlocksChanged();
} else if (!blocks.empty() && !blocks.back()->messages.empty()) {
blocks.back()->messages.back()->nextInBlocksRemoved();
}
}
void History::cacheTopPromoted(bool promoted) {
if (isTopPromoted() == promoted) {
return;
} else if (promoted) {
_flags |= Flag::IsTopPromoted;
} else {
_flags &= ~Flag::IsTopPromoted;
}
updateChatListSortPosition();
updateChatListEntry();
if (!isTopPromoted()) {
updateChatListExistence();
}
}
bool History::isTopPromoted() const {
return (_flags & Flag::IsTopPromoted);
}
void History::translateOfferFrom(LanguageId id) {
if (!id) {
if (translatedTo()) {
_translation->offerFrom(id);
} else if (_translation) {
_translation = nullptr;
session().changes().historyUpdated(
this,
UpdateFlag::TranslateFrom);
}
} else if (!_translation) {
_translation = std::make_unique<HistoryTranslation>(this, id);
} else {
_translation->offerFrom(id);
}
}
LanguageId History::translateOfferedFrom() const {
return _translation ? _translation->offeredFrom() : LanguageId();
}
void History::translateTo(LanguageId id) {
if (!_translation) {
return;
} else if (!id && !translateOfferedFrom()) {
_translation = nullptr;
session().changes().historyUpdated(this, UpdateFlag::TranslatedTo);
} else {
_translation->translateTo(id);
}
}
LanguageId History::translatedTo() const {
return _translation ? _translation->translatedTo() : LanguageId();
}
HistoryTranslation *History::translation() const {
return _translation.get();
}
HistoryBlock::HistoryBlock(not_null<History*> history)
: _history(history) {
}
int HistoryBlock::resizeGetHeight(int newWidth, ResizeRequest request) {
auto y = 0;
if (request == ResizeRequest::ReinitAll) {
for (const auto &message : messages) {
message->setY(y);
message->initDimensions();
y += message->resizeGetHeight(newWidth);
}
} else if (request == ResizeRequest::ResizeAll) {
for (const auto &message : messages) {
message->setY(y);
y += message->resizeGetHeight(newWidth);
}
} else {
for (const auto &message : messages) {
message->setY(y);
y += message->pendingResize()
? message->resizeGetHeight(newWidth)
: message->height();
}
}
_height = y;
return _height;
}
void HistoryBlock::remove(not_null<Element*> view) {
Expects(view->block() == this);
_history->mainViewRemoved(this, view);
const auto blockIndex = indexInHistory();
const auto itemIndex = view->indexInBlock();
const auto item = view->data();
item->clearMainView();
messages.erase(messages.begin() + itemIndex);
for (auto i = itemIndex, l = int(messages.size()); i < l; ++i) {
messages[i]->setIndexInBlock(i);
}
if (messages.empty()) {
// Deletes this.
_history->removeBlock(this);
} else if (itemIndex < messages.size()) {
messages[itemIndex]->previousInBlocksChanged();
} else if (blockIndex + 1 < _history->blocks.size()) {
_history->blocks[blockIndex + 1]->messages.front()->previousInBlocksChanged();
} else if (!_history->blocks.empty() && !_history->blocks.back()->messages.empty()) {
_history->blocks.back()->messages.back()->nextInBlocksRemoved();
}
}
void HistoryBlock::refreshView(not_null<Element*> view) {
Expects(view->block() == this);
const auto item = view->data();
auto refreshed = item->createView(
_history->delegateMixin()->delegate(),
view);
auto blockIndex = indexInHistory();
auto itemIndex = view->indexInBlock();
_history->viewReplaced(view, refreshed.get());
messages[itemIndex] = std::move(refreshed);
messages[itemIndex]->attachToBlock(this, itemIndex);
if (itemIndex + 1 < messages.size()) {
messages[itemIndex + 1]->previousInBlocksChanged();
} else if (blockIndex + 1 < _history->blocks.size()) {
_history->blocks[blockIndex + 1]->messages.front()->previousInBlocksChanged();
} else if (!_history->blocks.empty() && !_history->blocks.back()->messages.empty()) {
_history->blocks.back()->messages.back()->nextInBlocksRemoved();
}
}
HistoryBlock::~HistoryBlock() = default;