214 lines
4.8 KiB
C++
214 lines
4.8 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_indexed_list.h"
|
|
|
|
#include "auth_session.h"
|
|
#include "data/data_session.h"
|
|
#include "history/history.h"
|
|
|
|
namespace Dialogs {
|
|
|
|
IndexedList::IndexedList(SortMode sortMode)
|
|
: _sortMode(sortMode)
|
|
, _list(sortMode)
|
|
, _empty(sortMode) {
|
|
}
|
|
|
|
RowsByLetter IndexedList::addToEnd(Key key) {
|
|
RowsByLetter result;
|
|
if (!_list.contains(key)) {
|
|
result.emplace(0, _list.addToEnd(key));
|
|
for (auto ch : key.entry()->chatsListFirstLetters()) {
|
|
auto j = _index.find(ch);
|
|
if (j == _index.cend()) {
|
|
j = _index.emplace(
|
|
ch,
|
|
std::make_unique<List>(_sortMode)).first;
|
|
}
|
|
result.emplace(ch, j->second->addToEnd(key));
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
Row *IndexedList::addByName(Key key) {
|
|
if (const auto row = _list.getRow(key)) {
|
|
return row;
|
|
}
|
|
|
|
Row *result = _list.addByName(key);
|
|
for (auto ch : key.entry()->chatsListFirstLetters()) {
|
|
auto j = _index.find(ch);
|
|
if (j == _index.cend()) {
|
|
j = _index.emplace(
|
|
ch,
|
|
std::make_unique<List>(_sortMode)).first;
|
|
}
|
|
j->second->addByName(key);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void IndexedList::adjustByPos(const RowsByLetter &links) {
|
|
for (auto [ch, row] : links) {
|
|
if (ch == QChar(0)) {
|
|
_list.adjustByPos(row);
|
|
} else {
|
|
if (auto it = _index.find(ch); it != _index.cend()) {
|
|
it->second->adjustByPos(row);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void IndexedList::moveToTop(Key key) {
|
|
if (_list.moveToTop(key)) {
|
|
for (auto ch : key.entry()->chatsListFirstLetters()) {
|
|
if (auto it = _index.find(ch); it != _index.cend()) {
|
|
it->second->moveToTop(key);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void IndexedList::movePinned(Row *row, int deltaSign) {
|
|
auto swapPinnedIndexWith = find(row);
|
|
Assert(swapPinnedIndexWith != cend());
|
|
if (deltaSign > 0) {
|
|
++swapPinnedIndexWith;
|
|
} else {
|
|
Assert(swapPinnedIndexWith != cbegin());
|
|
--swapPinnedIndexWith;
|
|
}
|
|
Auth().data().reorderTwoPinnedDialogs(
|
|
row->key(),
|
|
(*swapPinnedIndexWith)->key());
|
|
}
|
|
|
|
void IndexedList::peerNameChanged(
|
|
not_null<PeerData*> peer,
|
|
const base::flat_set<QChar> &oldLetters) {
|
|
Expects(_sortMode != SortMode::Date);
|
|
|
|
if (const auto history = App::historyLoaded(peer)) {
|
|
if (_sortMode == SortMode::Name) {
|
|
adjustByName(history, oldLetters);
|
|
} else {
|
|
adjustNames(Dialogs::Mode::All, history, oldLetters);
|
|
}
|
|
}
|
|
}
|
|
|
|
void IndexedList::peerNameChanged(
|
|
Mode list,
|
|
not_null<PeerData*> peer,
|
|
const base::flat_set<QChar> &oldLetters) {
|
|
Expects(_sortMode == SortMode::Date);
|
|
|
|
if (const auto history = App::historyLoaded(peer)) {
|
|
adjustNames(list, history, oldLetters);
|
|
}
|
|
}
|
|
|
|
void IndexedList::adjustByName(
|
|
Key key,
|
|
const base::flat_set<QChar> &oldLetters) {
|
|
const auto mainRow = _list.adjustByName(key);
|
|
if (!mainRow) return;
|
|
|
|
auto toRemove = oldLetters;
|
|
auto toAdd = base::flat_set<QChar>();
|
|
for (auto ch : key.entry()->chatsListFirstLetters()) {
|
|
auto j = toRemove.find(ch);
|
|
if (j == toRemove.cend()) {
|
|
toAdd.insert(ch);
|
|
} else {
|
|
toRemove.erase(j);
|
|
if (auto it = _index.find(ch); it != _index.cend()) {
|
|
it->second->adjustByName(key);
|
|
}
|
|
}
|
|
}
|
|
for (auto ch : toRemove) {
|
|
if (auto it = _index.find(ch); it != _index.cend()) {
|
|
it->second->del(key, mainRow);
|
|
}
|
|
}
|
|
if (!toAdd.empty()) {
|
|
for (auto ch : toAdd) {
|
|
auto j = _index.find(ch);
|
|
if (j == _index.cend()) {
|
|
j = _index.emplace(
|
|
ch,
|
|
std::make_unique<List>(_sortMode)).first;
|
|
}
|
|
j->second->addByName(key);
|
|
}
|
|
}
|
|
}
|
|
|
|
void IndexedList::adjustNames(
|
|
Mode list,
|
|
not_null<History*> history,
|
|
const base::flat_set<QChar> &oldLetters) {
|
|
const auto key = Dialogs::Key(history);
|
|
auto mainRow = _list.getRow(key);
|
|
if (!mainRow) return;
|
|
|
|
auto toRemove = oldLetters;
|
|
auto toAdd = base::flat_set<QChar>();
|
|
for (auto ch : key.entry()->chatsListFirstLetters()) {
|
|
auto j = toRemove.find(ch);
|
|
if (j == toRemove.cend()) {
|
|
toAdd.insert(ch);
|
|
} else {
|
|
toRemove.erase(j);
|
|
}
|
|
}
|
|
for (auto ch : toRemove) {
|
|
if (_sortMode == SortMode::Date) {
|
|
history->removeChatListEntryByLetter(list, ch);
|
|
}
|
|
if (auto it = _index.find(ch); it != _index.cend()) {
|
|
it->second->del(key, mainRow);
|
|
}
|
|
}
|
|
for (auto ch : toAdd) {
|
|
auto j = _index.find(ch);
|
|
if (j == _index.cend()) {
|
|
j = _index.emplace(
|
|
ch,
|
|
std::make_unique<List>(_sortMode)).first;
|
|
}
|
|
auto row = j->second->addToEnd(key);
|
|
if (_sortMode == SortMode::Date) {
|
|
history->addChatListEntryByLetter(list, ch, row);
|
|
}
|
|
}
|
|
}
|
|
|
|
void IndexedList::del(Key key, Row *replacedBy) {
|
|
if (_list.del(key, replacedBy)) {
|
|
for (auto ch : key.entry()->chatsListFirstLetters()) {
|
|
if (auto it = _index.find(ch); it != _index.cend()) {
|
|
it->second->del(key, replacedBy);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void IndexedList::clear() {
|
|
_index.clear();
|
|
}
|
|
|
|
IndexedList::~IndexedList() {
|
|
clear();
|
|
}
|
|
|
|
} // namespace Dialogs
|