2022-09-20 18:12:30 +00:00
|
|
|
/*
|
|
|
|
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_forum_topic.h"
|
|
|
|
|
|
|
|
#include "data/data_channel.h"
|
|
|
|
#include "data/data_forum.h"
|
|
|
|
#include "data/data_session.h"
|
2022-09-27 16:52:35 +00:00
|
|
|
#include "data/stickers/data_custom_emoji.h"
|
2022-09-20 18:12:30 +00:00
|
|
|
#include "dialogs/dialogs_main_list.h"
|
2022-09-29 10:33:17 +00:00
|
|
|
#include "dialogs/ui/dialogs_layout.h"
|
2022-09-20 18:12:30 +00:00
|
|
|
#include "core/application.h"
|
|
|
|
#include "core/core_settings.h"
|
|
|
|
#include "history/history.h"
|
|
|
|
#include "history/history_item.h"
|
2022-09-27 16:52:35 +00:00
|
|
|
#include "ui/painter.h"
|
2022-09-29 10:33:17 +00:00
|
|
|
#include "styles/style_dialogs.h"
|
2022-09-20 18:12:30 +00:00
|
|
|
|
|
|
|
namespace Data {
|
|
|
|
|
2022-09-27 16:52:35 +00:00
|
|
|
ForumTopic::ForumTopic(not_null<History*> history, MsgId rootId)
|
|
|
|
: Entry(&history->owner(), Type::ForumTopic)
|
|
|
|
, _history(history)
|
|
|
|
, _list(forum()->topicsList())
|
2022-09-20 18:12:30 +00:00
|
|
|
, _rootId(rootId) {
|
|
|
|
}
|
|
|
|
|
2022-09-27 12:05:47 +00:00
|
|
|
not_null<ChannelData*> ForumTopic::channel() const {
|
2022-09-27 16:52:35 +00:00
|
|
|
return _history->peer->asChannel();
|
2022-09-27 12:05:47 +00:00
|
|
|
}
|
|
|
|
|
2022-09-27 16:52:35 +00:00
|
|
|
not_null<History*> ForumTopic::history() const {
|
|
|
|
return _history;
|
|
|
|
}
|
|
|
|
|
|
|
|
not_null<Forum*> ForumTopic::forum() const {
|
|
|
|
return channel()->forum();
|
2022-09-20 18:12:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MsgId ForumTopic::rootId() const {
|
|
|
|
return _rootId;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::applyTopic(const MTPForumTopic &topic) {
|
|
|
|
Expects(_rootId == topic.data().vid().v);
|
|
|
|
|
|
|
|
const auto &data = topic.data();
|
2022-09-23 19:21:31 +00:00
|
|
|
applyTitle(qs(data.vtitle()));
|
2022-09-27 12:05:47 +00:00
|
|
|
if (const auto iconId = data.vicon_emoji_id()) {
|
|
|
|
applyIconId(iconId->v);
|
|
|
|
} else {
|
|
|
|
applyIconId(0);
|
|
|
|
}
|
2022-09-20 18:12:30 +00:00
|
|
|
|
|
|
|
const auto pinned = _list->pinned();
|
2022-09-27 06:25:26 +00:00
|
|
|
#if 0 // #TODO forum pinned
|
2022-09-20 18:12:30 +00:00
|
|
|
if (data.is_pinned()) {
|
|
|
|
pinned->addPinned(Dialogs::Key(this));
|
|
|
|
} else {
|
|
|
|
pinned->setPinned(Dialogs::Key(this), false);
|
|
|
|
}
|
2022-09-27 06:25:26 +00:00
|
|
|
#endif
|
2022-09-20 18:12:30 +00:00
|
|
|
|
|
|
|
applyTopicFields(
|
|
|
|
data.vunread_count().v,
|
|
|
|
data.vread_inbox_max_id().v,
|
|
|
|
data.vread_outbox_max_id().v);
|
|
|
|
applyTopicTopMessage(data.vtop_message().v);
|
2022-09-27 06:25:26 +00:00
|
|
|
#if 0 // #TODO forum unread mark
|
|
|
|
setUnreadMark(data.is_unread_mark());
|
|
|
|
#endif
|
2022-09-20 18:12:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::indexTitleParts() {
|
|
|
|
_titleWords.clear();
|
|
|
|
_titleFirstLetters.clear();
|
|
|
|
auto toIndexList = QStringList();
|
|
|
|
auto appendToIndex = [&](const QString &value) {
|
|
|
|
if (!value.isEmpty()) {
|
|
|
|
toIndexList.push_back(TextUtilities::RemoveAccents(value));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
appendToIndex(_title);
|
|
|
|
const auto appendTranslit = !toIndexList.isEmpty()
|
|
|
|
&& cRussianLetters().match(toIndexList.front()).hasMatch();
|
|
|
|
if (appendTranslit) {
|
|
|
|
appendToIndex(translitRusEng(toIndexList.front()));
|
|
|
|
}
|
|
|
|
auto toIndex = toIndexList.join(' ');
|
|
|
|
toIndex += ' ' + rusKeyboardLayoutSwitch(toIndex);
|
|
|
|
|
|
|
|
const auto namesList = TextUtilities::PrepareSearchWords(toIndex);
|
|
|
|
for (const auto &name : namesList) {
|
|
|
|
_titleWords.insert(name);
|
|
|
|
_titleFirstLetters.insert(name[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int ForumTopic::chatListNameVersion() const {
|
|
|
|
return _titleVersion;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::applyTopicFields(
|
|
|
|
int unreadCount,
|
|
|
|
MsgId maxInboxRead,
|
|
|
|
MsgId maxOutboxRead) {
|
|
|
|
if (maxInboxRead + 1 >= _inboxReadBefore.value_or(1)) {
|
|
|
|
setUnreadCount(unreadCount);
|
|
|
|
setInboxReadTill(maxInboxRead);
|
|
|
|
}
|
|
|
|
setOutboxReadTill(maxOutboxRead);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::applyTopicTopMessage(MsgId topMessageId) {
|
|
|
|
if (topMessageId) {
|
2022-09-27 16:52:35 +00:00
|
|
|
const auto itemId = FullMsgId(_history->peer->id, topMessageId);
|
2022-09-20 18:12:30 +00:00
|
|
|
if (const auto item = owner().message(itemId)) {
|
|
|
|
setLastServerMessage(item);
|
|
|
|
} else {
|
|
|
|
setLastServerMessage(nullptr);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
setLastServerMessage(nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::setLastServerMessage(HistoryItem *item) {
|
|
|
|
_lastServerMessage = item;
|
|
|
|
if (_lastMessage
|
|
|
|
&& *_lastMessage
|
|
|
|
&& !(*_lastMessage)->isRegular()
|
|
|
|
&& (!item || (*_lastMessage)->date() > item->date())) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setLastMessage(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::setLastMessage(HistoryItem *item) {
|
|
|
|
if (_lastMessage && *_lastMessage == item) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_lastMessage = item;
|
|
|
|
if (!item || item->isRegular()) {
|
|
|
|
_lastServerMessage = item;
|
|
|
|
}
|
|
|
|
setChatListMessage(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::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());
|
|
|
|
|
|
|
|
#if 0 // #TODO forum
|
|
|
|
// If we have a single message from a group, request the full album.
|
|
|
|
if (hasOrphanMediaGroupPart()
|
|
|
|
&& !item->toPreview({
|
|
|
|
.hideSender = true,
|
|
|
|
.hideCaption = true }).images.empty()) {
|
|
|
|
owner().histories().requestGroupAround(item);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else if (!_chatListMessage || *_chatListMessage) {
|
|
|
|
_chatListMessage = nullptr;
|
|
|
|
updateChatListEntry();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::setInboxReadTill(MsgId upTo) {
|
|
|
|
if (_inboxReadBefore) {
|
|
|
|
accumulate_max(*_inboxReadBefore, upTo + 1);
|
|
|
|
} else {
|
|
|
|
_inboxReadBefore = upTo + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::setOutboxReadTill(MsgId upTo) {
|
|
|
|
if (_outboxReadBefore) {
|
|
|
|
accumulate_max(*_outboxReadBefore, upTo + 1);
|
|
|
|
} else {
|
|
|
|
_outboxReadBefore = upTo + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::loadUserpic() {
|
2022-09-27 16:52:35 +00:00
|
|
|
if (_icon) {
|
|
|
|
[[maybe_unused]] const auto preload = _icon->ready();
|
|
|
|
}
|
2022-09-20 18:12:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::paintUserpic(
|
|
|
|
Painter &p,
|
|
|
|
std::shared_ptr<Data::CloudImageView> &view,
|
2022-09-29 10:33:17 +00:00
|
|
|
const Dialogs::Ui::PaintContext &context) const {
|
2022-09-27 16:52:35 +00:00
|
|
|
if (_icon) {
|
2022-09-29 10:33:17 +00:00
|
|
|
const auto &st = context.st;
|
2022-09-27 16:52:35 +00:00
|
|
|
_icon->paint(p, {
|
|
|
|
.preview = st::windowBgOver->c,
|
2022-09-29 10:33:17 +00:00
|
|
|
.now = context.now,
|
|
|
|
.position = QPoint(st->padding.left(), st->padding.top()),
|
|
|
|
.paused = context.paused,
|
2022-09-27 16:52:35 +00:00
|
|
|
});
|
|
|
|
}
|
2022-09-20 18:12:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::requestChatListMessage() {
|
|
|
|
if (!chatListMessageKnown()) {
|
|
|
|
// #TODO forum
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TimeId ForumTopic::adjustedChatListTimeId() const {
|
2022-09-26 13:37:32 +00:00
|
|
|
if (isGeneral()) {
|
|
|
|
return TimeId(1);
|
|
|
|
}
|
2022-09-20 18:12:30 +00:00
|
|
|
const auto result = chatListTimeId();
|
|
|
|
#if 0 // #TODO forum
|
|
|
|
if (const auto draft = cloudDraft()) {
|
|
|
|
if (!Data::draftIsNull(draft) && !session().supportMode()) {
|
|
|
|
return std::max(result, draft->date);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ForumTopic::fixedOnTopIndex() const {
|
2022-09-23 19:21:31 +00:00
|
|
|
return 0;
|
2022-09-20 18:12:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::shouldBeInChatList() const {
|
|
|
|
return isPinnedDialog(FilterId())
|
|
|
|
|| !lastMessageKnown()
|
|
|
|
|| (lastMessage() != nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
HistoryItem *ForumTopic::lastMessage() const {
|
|
|
|
return _lastMessage.value_or(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::lastMessageKnown() const {
|
|
|
|
return _lastMessage.has_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
HistoryItem *ForumTopic::lastServerMessage() const {
|
|
|
|
return _lastServerMessage.value_or(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::lastServerMessageKnown() const {
|
|
|
|
return _lastServerMessage.has_value();
|
|
|
|
}
|
|
|
|
|
2022-09-27 12:05:47 +00:00
|
|
|
QString ForumTopic::title() const {
|
|
|
|
return _title;
|
|
|
|
}
|
|
|
|
|
2022-09-23 19:21:31 +00:00
|
|
|
void ForumTopic::applyTitle(const QString &title) {
|
2022-09-26 13:37:32 +00:00
|
|
|
if (_title == title || (isGeneral() && !_title.isEmpty())) {
|
2022-09-23 19:21:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2022-09-26 13:37:32 +00:00
|
|
|
_title = isGeneral() ? "General! Topic." : title; // #TODO forum lang
|
2022-09-23 19:21:31 +00:00
|
|
|
++_titleVersion;
|
|
|
|
indexTitleParts();
|
|
|
|
updateChatListEntry();
|
|
|
|
}
|
|
|
|
|
2022-09-27 12:05:47 +00:00
|
|
|
DocumentId ForumTopic::iconId() const {
|
|
|
|
return _iconId;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::applyIconId(DocumentId iconId) {
|
2022-09-27 16:52:35 +00:00
|
|
|
if (_iconId != iconId) {
|
|
|
|
_iconId = iconId;
|
|
|
|
_icon = iconId
|
|
|
|
? _history->owner().customEmojiManager().create(
|
|
|
|
_iconId,
|
|
|
|
[=] { updateChatListEntry(); },
|
2022-09-29 10:33:17 +00:00
|
|
|
Data::CustomEmojiManager::SizeTag::Normal)
|
2022-09-27 16:52:35 +00:00
|
|
|
: nullptr;
|
|
|
|
}
|
2022-09-27 12:05:47 +00:00
|
|
|
updateChatListEntry();
|
|
|
|
}
|
|
|
|
|
2022-09-23 19:21:31 +00:00
|
|
|
void ForumTopic::applyItemAdded(not_null<HistoryItem*> item) {
|
|
|
|
setLastMessage(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::applyItemRemoved(MsgId id) {
|
|
|
|
if (const auto lastItem = lastMessage()) {
|
|
|
|
if (lastItem->id == id) {
|
|
|
|
_lastMessage = std::nullopt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (const auto lastServerItem = lastServerMessage()) {
|
|
|
|
if (lastServerItem->id == id) {
|
|
|
|
_lastServerMessage = std::nullopt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (const auto chatListItem = _chatListMessage.value_or(nullptr)) {
|
|
|
|
if (chatListItem->id == id) {
|
|
|
|
_chatListMessage = std::nullopt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-20 18:12:30 +00:00
|
|
|
int ForumTopic::unreadCount() const {
|
|
|
|
return _unreadCount ? *_unreadCount : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ForumTopic::unreadCountForBadge() const {
|
|
|
|
const auto result = unreadCount();
|
|
|
|
return (!result && unreadMark()) ? 1 : result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::unreadCountKnown() const {
|
|
|
|
return _unreadCount.has_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::setUnreadCount(int newUnreadCount) {
|
|
|
|
if (_unreadCount == newUnreadCount) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const auto wasForBadge = (unreadCountForBadge() > 0);
|
|
|
|
const auto notifier = unreadStateChangeNotifier(true);
|
|
|
|
_unreadCount = newUnreadCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForumTopic::setUnreadMark(bool unread) {
|
|
|
|
if (_unreadMark == unread) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const auto noUnreadMessages = !unreadCount();
|
|
|
|
const auto refresher = gsl::finally([&] {
|
|
|
|
if (inChatList() && noUnreadMessages) {
|
|
|
|
updateChatListEntry();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
const auto notifier = unreadStateChangeNotifier(noUnreadMessages);
|
|
|
|
_unreadMark = unread;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::unreadMark() const {
|
|
|
|
return _unreadMark;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ForumTopic::chatListUnreadCount() const {
|
|
|
|
const auto state = chatListUnreadState();
|
|
|
|
return state.marks
|
|
|
|
+ (Core::App().settings().countUnreadMessages()
|
|
|
|
? state.messages
|
|
|
|
: state.chats);
|
|
|
|
}
|
|
|
|
|
|
|
|
Dialogs::UnreadState ForumTopic::chatListUnreadState() const {
|
|
|
|
auto result = Dialogs::UnreadState();
|
|
|
|
const auto count = _unreadCount.value_or(0);
|
|
|
|
const auto mark = !count && _unreadMark;
|
2022-09-27 16:52:35 +00:00
|
|
|
const auto muted = _history->mute();
|
2022-09-20 18:12:30 +00:00
|
|
|
result.messages = count;
|
|
|
|
result.messagesMuted = muted ? count : 0;
|
|
|
|
result.chats = count ? 1 : 0;
|
|
|
|
result.chatsMuted = (count && muted) ? 1 : 0;
|
|
|
|
result.marks = mark ? 1 : 0;
|
|
|
|
result.marksMuted = (mark && muted) ? 1 : 0;
|
|
|
|
result.known = _unreadCount.has_value();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::chatListUnreadMark() const {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::chatListMutedBadge() const {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
HistoryItem *ForumTopic::chatListMessage() const {
|
|
|
|
return _lastMessage.value_or(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ForumTopic::chatListMessageKnown() const {
|
|
|
|
return _lastMessage.has_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString &ForumTopic::chatListName() const {
|
|
|
|
return _title;
|
|
|
|
}
|
|
|
|
|
|
|
|
const base::flat_set<QString> &ForumTopic::chatListNameWords() const {
|
|
|
|
return _titleWords;
|
|
|
|
}
|
|
|
|
|
|
|
|
const base::flat_set<QChar> &ForumTopic::chatListFirstLetters() const {
|
|
|
|
return _titleFirstLetters;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString &ForumTopic::chatListNameSortKey() const {
|
|
|
|
static const auto empty = QString();
|
|
|
|
return empty;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Data
|