2
0
mirror of https://github.com/telegramdesktop/tdesktop synced 2025-08-23 18:57:12 +00:00

218 lines
6.1 KiB
C
Raw Normal View History

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "base/flat_map.h"
#include "base/weak_ptr.h"
#include "base/flags.h"
2024-11-27 14:27:31 +03:00
#include "dialogs/dialogs_common.h"
#include "ui/unread_badge.h"
class HistoryItem;
2024-11-27 14:27:31 +03:00
class History;
class UserData;
2019-07-24 13:45:24 +02:00
namespace Main {
class Session;
} // namespace Main
namespace Data {
class Session;
class Forum;
class Folder;
2022-09-20 22:12:30 +04:00
class ForumTopic;
class SavedSublist;
2025-05-13 18:18:24 +04:00
class SavedMessages;
2024-11-27 14:27:31 +03:00
class Thread;
} // namespace Data
namespace Ui {
struct PeerUserpicView;
} // namespace Ui
namespace Dialogs::Ui {
using namespace ::Ui;
struct PaintContext;
} // namespace Dialogs::Ui
namespace Dialogs {
2024-11-27 14:27:31 +03:00
struct UnreadState;
class Row;
class IndexedList;
class MainList;
2020-02-07 13:43:12 +04:00
[[nodiscard]] BadgesState BadgesForUnread(
const UnreadState &state,
CountInBadge count = CountInBadge::Default,
IncludeInBadge include = IncludeInBadge::Default);
class Entry : public base::has_weak_ptr {
public:
2022-09-20 22:12:30 +04:00
enum class Type : uchar {
2020-06-12 18:09:04 +04:00
History,
Folder,
2022-09-20 22:12:30 +04:00
ForumTopic,
SavedSublist,
2020-06-12 18:09:04 +04:00
};
Entry(not_null<Data::Session*> owner, Type type);
virtual ~Entry();
2020-02-07 13:43:12 +04:00
[[nodiscard]] Data::Session &owner() const;
[[nodiscard]] Main::Session &session() const;
2020-06-12 18:09:04 +04:00
History *asHistory();
Data::Forum *asForum();
2020-06-12 18:09:04 +04:00
Data::Folder *asFolder();
Data::Thread *asThread();
Data::ForumTopic *asTopic();
Data::SavedSublist *asSublist();
2020-06-12 18:09:04 +04:00
const History *asHistory() const;
const Data::Forum *asForum() const;
const Data::Folder *asFolder() const;
const Data::Thread *asThread() const;
const Data::ForumTopic *asTopic() const;
const Data::SavedSublist *asSublist() const;
PositionChange adjustByPosInChatList(
FilterId filterId,
not_null<MainList*> list);
2020-02-07 13:43:12 +04:00
[[nodiscard]] bool inChatList(FilterId filterId = 0) const {
return _chatListLinks.contains(filterId);
}
2020-06-12 18:09:04 +04:00
RowsByLetter *chatListLinks(FilterId filterId);
const RowsByLetter *chatListLinks(FilterId filterId) const;
2020-02-07 13:43:12 +04:00
[[nodiscard]] int posInChatList(FilterId filterId) const;
not_null<Row*> addToChatList(
FilterId filterId,
not_null<MainList*> list);
void setColorIndexForFilterId(FilterId, std::optional<uint8>);
void removeFromChatList(
FilterId filterId,
not_null<MainList*> list);
2020-02-07 13:43:12 +04:00
void removeChatListEntryByLetter(FilterId filterId, QChar letter);
void addChatListEntryByLetter(
2020-02-07 13:43:12 +04:00
FilterId filterId,
QChar letter,
not_null<Row*> row);
2020-06-12 18:09:04 +04:00
void updateChatListEntry();
void updateChatListEntryPostponed();
void updateChatListEntryHeight();
2020-03-17 17:04:30 +04:00
[[nodiscard]] bool isPinnedDialog(FilterId filterId) const {
return lookupPinnedIndex(filterId) != 0;
}
2020-03-17 17:04:30 +04:00
void cachePinnedIndex(FilterId filterId, int index);
[[nodiscard]] uint64 sortKeyInChatList(FilterId filterId) const {
return filterId
? computeSortPosition(filterId)
: _sortKeyInChatList;
}
void updateChatListSortPosition();
2019-01-15 15:57:45 +04:00
void setChatListTimeId(TimeId date);
2018-01-29 20:13:24 +03:00
virtual void updateChatListExistence();
bool needUpdateInChatList() const;
[[nodiscard]] virtual TimeId adjustedChatListTimeId() const;
[[nodiscard]] virtual int fixedOnTopIndex() const = 0;
2019-04-18 15:31:30 +04:00
static constexpr auto kArchiveFixOnTopIndex = 1;
2020-04-24 14:31:28 +04:00
static constexpr auto kTopPromotionFixOnTopIndex = 2;
2019-04-18 15:31:30 +04:00
[[nodiscard]] virtual bool shouldBeInChatList() const = 0;
[[nodiscard]] virtual UnreadState chatListUnreadState() const = 0;
[[nodiscard]] virtual BadgesState chatListBadgesState() const = 0;
[[nodiscard]] virtual HistoryItem *chatListMessage() const = 0;
[[nodiscard]] virtual bool chatListMessageKnown() const = 0;
[[nodiscard]] virtual const QString &chatListName() const = 0;
[[nodiscard]] virtual const QString &chatListNameSortKey() const = 0;
[[nodiscard]] virtual int chatListNameVersion() const = 0;
[[nodiscard]] virtual auto chatListNameWords() const
-> const base::flat_set<QString> & = 0;
[[nodiscard]] virtual auto chatListFirstLetters() const
-> const base::flat_set<QChar> & = 0;
[[nodiscard]] virtual bool folderKnown() const {
return true;
}
[[nodiscard]] virtual Data::Folder *folder() const {
return nullptr;
}
virtual void chatListPreloadData() = 0;
virtual void paintUserpic(
Painter &p,
Ui::PeerUserpicView &view,
const Ui::PaintContext &context) const = 0;
[[nodiscard]] TimeId chatListTimeId() const {
2019-01-15 15:57:45 +04:00
return _timeId;
}
2022-08-09 14:12:19 +03:00
[[nodiscard]] const Ui::Text::String &chatListNameText() const;
[[nodiscard]] Ui::PeerBadge &chatListPeerBadge() const {
return _chatListPeerBadge;
}
2022-08-09 14:12:19 +03:00
[[nodiscard]] bool hasChatsFilterTags(FilterId exclude) const;
2019-04-22 18:22:39 +04:00
protected:
void notifyUnreadStateChange(const UnreadState &wasState);
2022-10-20 18:57:15 +03:00
inline auto unreadStateChangeNotifier(bool required);
2019-04-22 18:22:39 +04:00
2020-03-17 17:04:30 +04:00
[[nodiscard]] int lookupPinnedIndex(FilterId filterId) const;
private:
enum class Flag : uchar {
IsThread = (1 << 0),
IsHistory = (1 << 1),
2025-05-13 18:18:24 +04:00
IsForumTopic = (1 << 2),
IsSavedSublist = (1 << 3),
UpdatePostponed = (1 << 4),
InUnreadChangeBlock = (1 << 5),
};
friend inline constexpr bool is_flag_type(Flag) { return true; }
using Flags = base::flags<Flag>;
virtual void changedChatListPinHook();
2021-12-02 11:51:54 +04:00
void pinnedIndexChanged(FilterId filterId, int was, int now);
2020-03-17 17:04:30 +04:00
[[nodiscard]] uint64 computeSortPosition(FilterId filterId) const;
void setChatListExistence(bool exists);
2020-02-07 13:43:12 +04:00
not_null<Row*> mainChatListLink(FilterId filterId) const;
Row *maybeMainChatListLink(FilterId filterId) const;
2020-06-12 18:09:04 +04:00
const not_null<Data::Session*> _owner;
2020-02-07 13:43:12 +04:00
base::flat_map<FilterId, RowsByLetter> _chatListLinks;
uint64 _sortKeyInChatList = 0;
uint64 _sortKeyByDate = 0;
2020-03-17 17:04:30 +04:00
base::flat_map<FilterId, int> _pinnedIndex;
base::flat_map<FilterId, uint8> _tagColors;
mutable Ui::PeerBadge _chatListPeerBadge;
2022-08-09 14:12:19 +03:00
mutable Ui::Text::String _chatListNameText;
mutable int _chatListNameVersion = 0;
2019-01-15 15:57:45 +04:00
TimeId _timeId = 0;
Flags _flags;
};
auto Entry::unreadStateChangeNotifier(bool required) {
Expects(!(_flags & Flag::InUnreadChangeBlock));
_flags |= Flag::InUnreadChangeBlock;
const auto notify = required && inChatList();
const auto wasState = notify ? chatListUnreadState() : UnreadState();
return gsl::finally([=, this] {
_flags &= ~Flag::InUnreadChangeBlock;
if (notify) {
Assert(inChatList());
notifyUnreadStateChange(wasState);
}
});
}
} // namespace Dialogs