Show last chats in archive dialog row.

This commit is contained in:
John Preston 2019-05-08 11:50:39 +03:00
parent 4f65d0469c
commit e55e46a0f0
4 changed files with 65 additions and 100 deletions

View File

@ -1202,11 +1202,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
"lng_archived_remove" = "Unarchive"; "lng_archived_remove" = "Unarchive";
"lng_archived_added" = "Chat archived.\nMuted chats will stay archived after new messages arrive."; "lng_archived_added" = "Chat archived.\nMuted chats will stay archived after new messages arrive.";
"lng_archived_removed" = "Chat restored from your archive."; "lng_archived_removed" = "Chat restored from your archive.";
"lng_archived_chats#one" = "{count} chat"; "lng_archived_last_list" = "{accumulated}, {chat}";
"lng_archived_chats#other" = "{count} chats"; "lng_archived_last#one" = "{chats} and {count} more chat";
"lng_archived_unread_two" = "{chat}, {second_chat}"; "lng_archived_last#other" = "{chats} and {count} more chats";
"lng_archived_unread#one" = "{chat}, {second_chat} and {count} more unread chat";
"lng_archived_unread#other" = "{chat}, {second_chat} and {count} more unread chats";
"lng_dialogs_text_with_from" = "{from_part} {message}"; "lng_dialogs_text_with_from" = "{from_part} {message}";
"lng_dialogs_text_from_wrapped" = "{from}:"; "lng_dialogs_text_from_wrapped" = "{from}:";

View File

@ -26,7 +26,7 @@ namespace Data {
namespace { namespace {
constexpr auto kLoadedChatsMinCount = 20; constexpr auto kLoadedChatsMinCount = 20;
constexpr auto kShowChatNamesCount = 2; constexpr auto kShowChatNamesCount = 8;
rpl::producer<int> PinnedDialogsInFolderMaxValue() { rpl::producer<int> PinnedDialogsInFolderMaxValue() {
return rpl::single( return rpl::single(
@ -60,7 +60,7 @@ Folder::Folder(not_null<Data::Session*> owner, FolderId id)
Notify::PeerUpdateViewer( Notify::PeerUpdateViewer(
Notify::PeerUpdate::Flag::NameChanged Notify::PeerUpdate::Flag::NameChanged
) | rpl::start_with_next([=](const Notify::PeerUpdate &update) { ) | rpl::start_with_next([=](const Notify::PeerUpdate &update) {
for (const auto history : _unreadHistoriesLast) { for (const auto history : _lastHistories) {
if (history->peer == update.peer) { if (history->peer == update.peer) {
++_chatListViewVersion; ++_chatListViewVersion;
updateChatListEntry(); updateChatListEntry();
@ -112,6 +112,7 @@ void Folder::registerOne(not_null<History*> history) {
updateChatListEntry(); updateChatListEntry();
} }
applyChatListMessage(history->chatListMessage()); applyChatListMessage(history->chatListMessage());
reorderLastHistories();
} }
void Folder::unregisterOne(not_null<History*> history) { void Folder::unregisterOne(not_null<History*> history) {
@ -124,6 +125,7 @@ void Folder::unregisterOne(not_null<History*> history) {
if (_chatListMessage && _chatListMessage->history() == history) { if (_chatListMessage && _chatListMessage->history() == history) {
computeChatListMessage(); computeChatListMessage();
} }
reorderLastHistories();
} }
void Folder::oneListMessageChanged(HistoryItem *from, HistoryItem *to) { void Folder::oneListMessageChanged(HistoryItem *from, HistoryItem *to) {
@ -131,10 +133,7 @@ void Folder::oneListMessageChanged(HistoryItem *from, HistoryItem *to) {
computeChatListMessage(); computeChatListMessage();
} }
if (from || to) { if (from || to) {
const auto history = from ? from->history() : to->history(); reorderLastHistories();
if (!history->chatListUnreadState().empty()) {
reorderUnreadHistories();
}
} }
} }
@ -171,49 +170,34 @@ void Folder::computeChatListMessage() {
updateChatListEntry(); updateChatListEntry();
} }
void Folder::addUnreadHistory(not_null<History*> history) { void Folder::reorderLastHistories() {
const auto i = ranges::find(_unreadHistories, history);
if (i == end(_unreadHistories)) {
_unreadHistories.push_back(history);
reorderUnreadHistories();
}
}
void Folder::removeUnreadHistory(not_null<History*> history) {
const auto i = ranges::find(_unreadHistories, history);
if (i != end(_unreadHistories)) {
_unreadHistories.erase(i);
reorderUnreadHistories();
}
}
void Folder::reorderUnreadHistories() {
// We want first kShowChatNamesCount histories, by last message date. // We want first kShowChatNamesCount histories, by last message date.
const auto predicate = [](not_null<History*> a, not_null<History*> b) { const auto pred = [](not_null<History*> a, not_null<History*> b) {
const auto aItem = a->chatListMessage(); const auto aItem = a->chatListMessage();
const auto bItem = b->chatListMessage(); const auto bItem = b->chatListMessage();
const auto aDate = aItem ? aItem->date() : TimeId(0); const auto aDate = aItem ? aItem->date() : TimeId(0);
const auto bDate = bItem ? bItem->date() : TimeId(0); const auto bDate = bItem ? bItem->date() : TimeId(0);
return aDate > bDate; return aDate > bDate;
}; };
if (size(_unreadHistories) <= kShowChatNamesCount) { _lastHistories.erase(_lastHistories.begin(), _lastHistories.end());
ranges::sort(_unreadHistories, predicate); _lastHistories.reserve(kShowChatNamesCount + 1);
if (!ranges::equal(_unreadHistories, _unreadHistoriesLast)) { auto &&histories = ranges::view::all(
_unreadHistoriesLast = _unreadHistories; *_chatsList.indexed()
) | ranges::view::transform([](not_null<Dialogs::Row*> row) {
return row->history();
}) | ranges::view::filter([](History *history) {
return (history != nullptr);
}) | ranges::view::transform([](History *history) {
return not_null<History*>(history);
});
for (const auto history : histories) {
const auto i = ranges::upper_bound(_lastHistories, history, pred);
if (size(_lastHistories) < kShowChatNamesCount
|| i != end(_lastHistories)) {
_lastHistories.insert(i, history);
} }
} else { if (size(_lastHistories) > kShowChatNamesCount) {
const auto till = begin(_unreadHistories) + kShowChatNamesCount - 1; _lastHistories.pop_back();
ranges::nth_element(_unreadHistories, till, predicate);
if constexpr (kShowChatNamesCount > 2) {
ranges::sort(begin(_unreadHistories), till, predicate);
}
auto &&head = ranges::view::all(
_unreadHistories
) | ranges::view::take_exactly(
kShowChatNamesCount
);
if (!ranges::equal(head, _unreadHistoriesLast)) {
_unreadHistoriesLast = head | ranges::to_vector;
} }
} }
++_chatListViewVersion; ++_chatListViewVersion;
@ -296,12 +280,8 @@ int Folder::chatsListSize() const {
_chatsList.loaded() ? 0 : _cloudChatsListSize); _chatsList.loaded() ? 0 : _cloudChatsListSize);
} }
int Folder::unreadHistoriesCount() const { const std::vector<not_null<History*>> &Folder::lastHistories() const {
return _unreadHistories.size(); return _lastHistories;
}
const std::vector<not_null<History*>> &Folder::lastUnreadHistories() const {
return _unreadHistoriesLast;
} }
uint32 Folder::chatListViewVersion() const { uint32 Folder::chatListViewVersion() const {
@ -380,14 +360,6 @@ void Folder::unreadStateChanged(
const Dialogs::Key &key, const Dialogs::Key &key,
const Dialogs::UnreadState &wasState, const Dialogs::UnreadState &wasState,
const Dialogs::UnreadState &nowState) { const Dialogs::UnreadState &nowState) {
if (const auto history = key.history()) {
if (!wasState.empty() && nowState.empty()) {
removeUnreadHistory(history);
} else if (wasState.empty() && !nowState.empty()) {
addUnreadHistory(history);
}
}
const auto updateCloudUnread = _cloudUnread.known && wasState.known; const auto updateCloudUnread = _cloudUnread.known && wasState.known;
const auto notify = _chatsList.loaded() || updateCloudUnread; const auto notify = _chatsList.loaded() || updateCloudUnread;
const auto notifier = unreadStateChangeNotifier(notify); const auto notifier = unreadStateChangeNotifier(notify);
@ -404,16 +376,6 @@ void Folder::unreadEntryChanged(
const Dialogs::Key &key, const Dialogs::Key &key,
const Dialogs::UnreadState &state, const Dialogs::UnreadState &state,
bool added) { bool added) {
if (const auto history = key.history()) {
if (!state.empty()) {
if (added) {
addUnreadHistory(history);
} else {
removeUnreadHistory(history);
}
}
}
const auto updateCloudUnread = _cloudUnread.known && state.known; const auto updateCloudUnread = _cloudUnread.known && state.known;
const auto notify = _chatsList.loaded() || updateCloudUnread; const auto notify = _chatsList.loaded() || updateCloudUnread;
const auto notifier = unreadStateChangeNotifier(notify); const auto notifier = unreadStateChangeNotifier(notify);

View File

@ -80,8 +80,7 @@ public:
void setCloudChatsListSize(int size); void setCloudChatsListSize(int size);
int chatsListSize() const; int chatsListSize() const;
int unreadHistoriesCount() const; const std::vector<not_null<History*>> &lastHistories() const;
const std::vector<not_null<History*>> &lastUnreadHistories() const;
uint32 chatListViewVersion() const; uint32 chatListViewVersion() const;
private: private:
@ -89,9 +88,7 @@ private:
bool applyChatListMessage(HistoryItem *item); bool applyChatListMessage(HistoryItem *item);
void computeChatListMessage(); void computeChatListMessage();
void addUnreadHistory(not_null<History*> history); void reorderLastHistories();
void removeUnreadHistory(not_null<History*> history);
void reorderUnreadHistories();
void finalizeCloudUnread(); void finalizeCloudUnread();
FolderId _id = 0; FolderId _id = 0;
@ -103,8 +100,7 @@ private:
Dialogs::UnreadState _cloudUnread; Dialogs::UnreadState _cloudUnread;
int _cloudChatsListSize = 0; int _cloudChatsListSize = 0;
std::vector<not_null<History*>> _unreadHistories; std::vector<not_null<History*>> _lastHistories;
std::vector<not_null<History*>> _unreadHistoriesLast;
HistoryItem *_chatListMessage = nullptr; HistoryItem *_chatListMessage = nullptr;
uint32 _chatListViewVersion = 0; uint32 _chatListViewVersion = 0;
//rpl::variable<MessagePosition> _unreadPosition; //rpl::variable<MessagePosition> _unreadPosition;

View File

@ -20,33 +20,42 @@ namespace Dialogs {
namespace { namespace {
QString ComposeFolderListEntryText(not_null<Data::Folder*> folder) { QString ComposeFolderListEntryText(not_null<Data::Folder*> folder) {
const auto &list = folder->lastUnreadHistories(); const auto &list = folder->lastHistories();
if (list.empty()) { if (list.empty()) {
const auto count = folder->chatsListSize(); return QString();
if (!count) {
return QString();
}
return lng_archived_chats(lt_count, count);
} }
const auto count = std::max( const auto count = std::max(
int(list.size()), int(list.size()),
folder->unreadHistoriesCount()); folder->chatsListSize());
if (list.size() == 1) {
return App::peerName(list[0]->peer); const auto throwAwayLastName = (list.size() > 1)
} else if (count == 2) { && (count == list.size() + 1);
return lng_archived_unread_two( auto &&peers = ranges::view::all(
lt_chat, list
App::peerName(list[0]->peer), ) | ranges::view::take(
lt_second_chat, list.size() - (throwAwayLastName ? 1 : 0)
App::peerName(list[1]->peer)); ) | ranges::view::transform([](not_null<History*> history) {
} return history->peer;
return lng_archived_unread( });
lt_count, const auto shown = int(peers.size());
count - 2, const auto accumulated = [&] {
lt_chat, Expects(shown > 0);
App::peerName(list[0]->peer),
lt_second_chat, auto i = peers.begin();
App::peerName(list[1]->peer)); auto result = App::peerName(*i);
for (++i; i != peers.end(); ++i) {
result = lng_archived_last_list(
lt_accumulated,
result,
lt_chat,
App::peerName(*i));
}
return result;
}();
return (shown < count)
? lng_archived_last(lt_count, (count - shown), lt_chats, accumulated)
: accumulated;
} }
} // namespace } // namespace