229 lines
5.7 KiB
C++
229 lines
5.7 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 "dialogs/dialogs_main_list.h"
|
|
|
|
#include "data/data_changes.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_chat_filters.h"
|
|
#include "main/main_session.h"
|
|
#include "history/history_unread_things.h"
|
|
#include "history/history.h"
|
|
|
|
namespace Dialogs {
|
|
|
|
MainList::MainList(
|
|
not_null<Main::Session*> session,
|
|
FilterId filterId,
|
|
rpl::producer<int> pinnedLimit)
|
|
: _filterId(filterId)
|
|
, _all(SortMode::Date, filterId)
|
|
, _pinned(filterId, 1) {
|
|
_unreadState.known = true;
|
|
|
|
std::move(
|
|
pinnedLimit
|
|
) | rpl::start_with_next([=](int limit) {
|
|
_pinned.setLimit(limit);
|
|
}, _lifetime);
|
|
|
|
session->changes().realtimeNameUpdates(
|
|
) | rpl::start_with_next([=](const Data::NameUpdate &update) {
|
|
_all.peerNameChanged(_filterId, update.peer, update.oldFirstLetters);
|
|
}, _lifetime);
|
|
}
|
|
|
|
bool MainList::empty() const {
|
|
return _all.empty();
|
|
}
|
|
|
|
bool MainList::loaded() const {
|
|
return _loaded;
|
|
}
|
|
|
|
void MainList::setLoaded(bool loaded) {
|
|
if (_loaded == loaded) {
|
|
return;
|
|
}
|
|
const auto recomputer = gsl::finally([&] {
|
|
recomputeFullListSize();
|
|
});
|
|
const auto notifier = unreadStateChangeNotifier(true);
|
|
_loaded = loaded;
|
|
}
|
|
|
|
void MainList::setAllAreMuted(bool allAreMuted) {
|
|
if (_allAreMuted == allAreMuted) {
|
|
return;
|
|
}
|
|
const auto notifier = unreadStateChangeNotifier(true);
|
|
_allAreMuted = allAreMuted;
|
|
}
|
|
|
|
void MainList::setCloudListSize(int size) {
|
|
if (_cloudListSize == size) {
|
|
return;
|
|
}
|
|
_cloudListSize = size;
|
|
recomputeFullListSize();
|
|
}
|
|
|
|
const rpl::variable<int> &MainList::fullSize() const {
|
|
return _fullListSize;
|
|
}
|
|
|
|
void MainList::clear() {
|
|
const auto recomputer = gsl::finally([&] {
|
|
recomputeFullListSize();
|
|
});
|
|
const auto notifier = unreadStateChangeNotifier(true);
|
|
_pinned.clear();
|
|
_all.clear();
|
|
_unreadState = UnreadState();
|
|
_cloudUnreadState = UnreadState();
|
|
_unreadState.known = true;
|
|
_cloudUnreadState.known = true;
|
|
_cloudListSize = 0;
|
|
}
|
|
|
|
RowsByLetter MainList::addEntry(Key key) {
|
|
const auto result = _all.addToEnd(key);
|
|
|
|
const auto unread = key.entry()->chatListUnreadState();
|
|
unreadEntryChanged(unread, true);
|
|
recomputeFullListSize();
|
|
|
|
return result;
|
|
}
|
|
|
|
void MainList::removeEntry(Key key) {
|
|
_all.remove(key);
|
|
|
|
const auto unread = key.entry()->chatListUnreadState();
|
|
unreadEntryChanged(unread, false);
|
|
recomputeFullListSize();
|
|
}
|
|
|
|
void MainList::recomputeFullListSize() {
|
|
_fullListSize = std::max(_all.size(), loaded() ? 0 : _cloudListSize);
|
|
}
|
|
|
|
void MainList::unreadStateChanged(
|
|
const UnreadState &wasState,
|
|
const UnreadState &nowState) {
|
|
const auto useClouded = _cloudUnreadState.known && !loaded();
|
|
const auto updateCloudUnread = _cloudUnreadState.known && wasState.known;
|
|
const auto notify = !useClouded || wasState.known;
|
|
const auto notifier = unreadStateChangeNotifier(notify);
|
|
_unreadState += nowState - wasState;
|
|
if (updateCloudUnread) {
|
|
Assert(nowState.known);
|
|
_cloudUnreadState += nowState - wasState;
|
|
finalizeCloudUnread();
|
|
}
|
|
}
|
|
|
|
void MainList::unreadEntryChanged(
|
|
const Dialogs::UnreadState &state,
|
|
bool added) {
|
|
if (!state.messages
|
|
&& !state.chats
|
|
&& !state.marks
|
|
&& !state.mentions
|
|
&& !state.reactions) {
|
|
return;
|
|
}
|
|
const auto updateCloudUnread = _cloudUnreadState.known && state.known;
|
|
const auto notify = !_cloudUnreadState.known || loaded() || state.known;
|
|
const auto notifier = unreadStateChangeNotifier(notify);
|
|
if (added) {
|
|
_unreadState += state;
|
|
} else {
|
|
_unreadState -= state;
|
|
}
|
|
if (updateCloudUnread) {
|
|
if (added) {
|
|
_cloudUnreadState += state;
|
|
} else {
|
|
_cloudUnreadState -= state;
|
|
}
|
|
finalizeCloudUnread();
|
|
}
|
|
}
|
|
|
|
void MainList::updateCloudUnread(const MTPDdialogFolder &data) {
|
|
const auto notifier = unreadStateChangeNotifier(!loaded());
|
|
|
|
_cloudUnreadState.messages = data.vunread_muted_messages_count().v
|
|
+ data.vunread_unmuted_messages_count().v;
|
|
_cloudUnreadState.chats = data.vunread_muted_peers_count().v
|
|
+ data.vunread_unmuted_peers_count().v;
|
|
finalizeCloudUnread();
|
|
|
|
_cloudUnreadState.known = true;
|
|
}
|
|
|
|
bool MainList::cloudUnreadKnown() const {
|
|
return _cloudUnreadState.known;
|
|
}
|
|
|
|
void MainList::finalizeCloudUnread() {
|
|
// Cloud state for archive folder always counts everything as muted.
|
|
_cloudUnreadState.messagesMuted = _cloudUnreadState.messages;
|
|
_cloudUnreadState.chatsMuted = _cloudUnreadState.chats;
|
|
|
|
// We don't know the real value of marked chats counts in cloud unread.
|
|
_cloudUnreadState.marksMuted = _cloudUnreadState.marks = 0;
|
|
}
|
|
|
|
UnreadState MainList::unreadState() const {
|
|
const auto useCloudState = _cloudUnreadState.known && !loaded();
|
|
auto result = useCloudState ? _cloudUnreadState : _unreadState;
|
|
|
|
// We don't know the real value of marked chats counts in cloud unread.
|
|
if (useCloudState) {
|
|
result.marks = _unreadState.marks;
|
|
result.marksMuted = _unreadState.marksMuted;
|
|
}
|
|
if (_allAreMuted) {
|
|
result.messagesMuted = result.messages;
|
|
result.chatsMuted = result.chats;
|
|
result.marksMuted = result.marks;
|
|
}
|
|
#ifdef Q_OS_WIN
|
|
[[maybe_unused]] volatile auto touch = 0
|
|
+ _unreadState.marks + _unreadState.marksMuted
|
|
+ _unreadState.messages + _unreadState.messagesMuted
|
|
+ _unreadState.chats + _unreadState.chatsMuted
|
|
+ _unreadState.reactions + _unreadState.reactionsMuted
|
|
+ _unreadState.mentions;
|
|
#endif // Q_OS_WIN
|
|
return result;
|
|
}
|
|
|
|
rpl::producer<UnreadState> MainList::unreadStateChanges() const {
|
|
return _unreadStateChanges.events();
|
|
}
|
|
|
|
not_null<IndexedList*> MainList::indexed() {
|
|
return &_all;
|
|
}
|
|
|
|
not_null<const IndexedList*> MainList::indexed() const {
|
|
return &_all;
|
|
}
|
|
|
|
not_null<PinnedList*> MainList::pinned() {
|
|
return &_pinned;
|
|
}
|
|
|
|
not_null<const PinnedList*> MainList::pinned() const {
|
|
return &_pinned;
|
|
}
|
|
|
|
} // namespace Dialogs
|