Read chat history while scrolling.

GitOrigin-RevId: 0795d97634a7420d2d3784304ae314cd075ac1d5
This commit is contained in:
levlam 2018-05-03 11:57:19 +03:00
parent 5b1b1273b4
commit 1b64400167
4 changed files with 42 additions and 44 deletions

View File

@ -140,6 +140,11 @@ class MessageId {
return ServerMessageId(narrow_cast<int32>(id >> SERVER_ID_SHIFT)); return ServerMessageId(narrow_cast<int32>(id >> SERVER_ID_SHIFT));
} }
// returns greatest server message id not bigger than this message id
MessageId get_prev_server_message_id() const {
return MessageId(id & ~FULL_TYPE_MASK);
}
// returns smallest server message id not less than this message id // returns smallest server message id not less than this message id
MessageId get_next_server_message_id() const { MessageId get_next_server_message_id() const {
return MessageId((id + FULL_TYPE_MASK) & ~FULL_TYPE_MASK); return MessageId((id + FULL_TYPE_MASK) & ~FULL_TYPE_MASK);

View File

@ -8862,6 +8862,16 @@ int32 MessagesManager::calc_new_unread_count(Dialog *d, MessageId max_message_id
} }
} }
void MessagesManager::repair_server_unread_count(DialogId dialog_id, int32 unread_count) {
LOG(INFO) << "Repair server unread count in " << dialog_id << " from " << unread_count;
create_actor<SleepActor>("RepairServerUnreadCountSleepActor", 0.5,
PromiseCreator::lambda([actor_id = actor_id(this), dialog_id](Result<Unit> result) {
send_closure(actor_id, &MessagesManager::send_get_dialog_query, dialog_id,
Promise<Unit>());
}))
.release();
}
void MessagesManager::read_history_inbox(DialogId dialog_id, MessageId max_message_id, int32 unread_count, void MessagesManager::read_history_inbox(DialogId dialog_id, MessageId max_message_id, int32 unread_count,
const char *source) { const char *source) {
if (td_->auth_manager_->is_bot()) { if (td_->auth_manager_->is_bot()) {
@ -8929,9 +8939,8 @@ void MessagesManager::read_history_inbox(DialogId dialog_id, MessageId max_messa
if (server_unread_count < 0) { if (server_unread_count < 0) {
server_unread_count = unread_count >= 0 ? unread_count : d->server_unread_count; server_unread_count = unread_count >= 0 ? unread_count : d->server_unread_count;
if (dialog_id.get_type() != DialogType::SecretChat && have_input_peer(dialog_id, AccessRights::Read)) { if (dialog_id.get_type() != DialogType::SecretChat && have_input_peer(dialog_id, AccessRights::Read)) {
LOG(INFO) << "Repair server unread count in " << d->dialog_id << " from " << server_unread_count;
d->need_repair_server_unread_count = true; d->need_repair_server_unread_count = true;
send_get_dialog_query(dialog_id, Auto()); repair_server_unread_count(dialog_id, server_unread_count);
} }
} }
if (local_unread_count < 0) { if (local_unread_count < 0) {
@ -11021,15 +11030,17 @@ void MessagesManager::on_get_dialogs(vector<tl_object_ptr<telegram_api::dialog>>
if (!G()->parameters().use_message_db || is_new || !d->is_last_read_inbox_message_id_inited || if (!G()->parameters().use_message_db || is_new || !d->is_last_read_inbox_message_id_inited ||
d->need_repair_server_unread_count) { d->need_repair_server_unread_count) {
if (d->need_repair_server_unread_count) {
LOG(INFO) << "Repaired server unread count in " << dialog_id << " from " << d->last_read_inbox_message_id << "/"
<< d->server_unread_count << " to " << read_inbox_max_message_id << "/" << dialog->unread_count_;
d->need_repair_server_unread_count = false;
on_dialog_updated(dialog_id, "repair dialog server unread count");
}
if (d->server_unread_count != dialog->unread_count_ || if (d->server_unread_count != dialog->unread_count_ ||
d->last_read_inbox_message_id.get() < read_inbox_max_message_id.get()) { d->last_read_inbox_message_id.get() < read_inbox_max_message_id.get()) {
set_dialog_last_read_inbox_message_id(d, read_inbox_max_message_id, dialog->unread_count_, set_dialog_last_read_inbox_message_id(d, read_inbox_max_message_id, dialog->unread_count_,
d->local_unread_count, true, "on_get_dialogs"); d->local_unread_count, true, "on_get_dialogs");
} }
if (d->need_repair_server_unread_count) {
d->need_repair_server_unread_count = false;
on_dialog_updated(dialog_id, "repair dialog server unread count");
}
} }
if (!G()->parameters().use_message_db || is_new || !d->is_last_read_outbox_message_id_inited) { if (!G()->parameters().use_message_db || is_new || !d->is_last_read_outbox_message_id_inited) {
@ -12866,8 +12877,8 @@ Status MessagesManager::view_messages(DialogId dialog_id, const vector<MessageId
bool need_read = force_read || d->is_opened; bool need_read = force_read || d->is_opened;
bool is_secret = dialog_id.get_type() == DialogType::SecretChat; bool is_secret = dialog_id.get_type() == DialogType::SecretChat;
MessageId max_incoming_message_id; MessageId max_incoming_message_id; // max remotely available viewed incoming message_id
MessageId max_message_id; MessageId max_message_id; // max server or local viewed message_id
vector<MessageId> read_content_message_ids; vector<MessageId> read_content_message_ids;
for (auto message_id : message_ids) { for (auto message_id : message_ids) {
auto message = get_message_force(d, message_id); auto message = get_message_force(d, message_id);
@ -12905,42 +12916,23 @@ Status MessagesManager::view_messages(DialogId dialog_id, const vector<MessageId
} }
if (need_read && max_message_id.get() > d->last_read_inbox_message_id.get()) { if (need_read && max_message_id.get() > d->last_read_inbox_message_id.get()) {
MessageId last_read_message_id = d->last_message_id; MessageId last_read_message_id = max_message_id;
if (!last_read_message_id.is_valid() || last_read_message_id.is_yet_unsent()) {
if (max_message_id.get() <= d->last_new_message_id.get()) { if (dialog_id.get_type() != DialogType::SecretChat) {
last_read_message_id = d->last_new_message_id; if (last_read_message_id.get_prev_server_message_id().get() >
} else { d->last_read_inbox_message_id.get_prev_server_message_id().get()) {
last_read_message_id = max_message_id; bool allow_error =
max_incoming_message_id.get() <= d->last_read_inbox_message_id.get_prev_server_message_id().get();
read_history_on_server(d->dialog_id, last_read_message_id.get_prev_server_message_id(), allow_error, 0);
}
} else {
if (last_read_message_id.get() > d->last_read_inbox_message_id.get()) {
bool allow_error = max_incoming_message_id.get() <= d->last_read_inbox_message_id.get();
read_history_on_server(d->dialog_id, last_read_message_id, allow_error, 0);
} }
} }
read_history_inbox(d->dialog_id, last_read_message_id, -1, "view_messages"); read_history_inbox(d->dialog_id, last_read_message_id, -1, "view_messages");
if (d->last_new_message_id.is_valid()) {
if (!d->is_last_read_inbox_message_id_inited) {
// don't know last read inbox message, read history on the server just in case
read_history_on_server(d->dialog_id, d->last_new_message_id, true, 0);
} else {
if (max_incoming_message_id.get() <= d->last_read_inbox_message_id.get()) {
MessagesConstIterator p(d, d->last_message_id);
while (*p != nullptr && ((*p)->is_outgoing || !((*p)->message_id.is_server() || is_secret)) &&
(*p)->message_id.get() > d->last_read_inbox_message_id.get()) {
--p;
}
if (*p != nullptr && !(*p)->is_outgoing && ((*p)->message_id.is_server() || is_secret)) {
max_incoming_message_id = (*p)->message_id;
}
}
if (max_incoming_message_id.get() > d->last_read_inbox_message_id.get()) {
LOG_IF(ERROR, d->server_unread_count + d->local_unread_count == 0) << "Nave no unread messages";
read_history_on_server(d->dialog_id, d->last_new_message_id, false,
0); // TODO can read messages only up to max_incoming_message_id
} else {
// can't find last incoming message, read history on the server just in case
read_history_on_server(d->dialog_id, d->last_new_message_id, true, 0);
}
}
}
} }
return Status::OK(); return Status::OK();
@ -23980,9 +23972,8 @@ void MessagesManager::fix_new_dialog(Dialog *d, unique_ptr<Message> &&last_datab
} }
if (d->need_repair_server_unread_count && have_input_peer(dialog_id, AccessRights::Read)) { if (d->need_repair_server_unread_count && have_input_peer(dialog_id, AccessRights::Read)) {
LOG(INFO) << "Repair server unread count in " << d->dialog_id << " from " << d->server_unread_count;
CHECK(dialog_id.get_type() != DialogType::SecretChat); CHECK(dialog_id.get_type() != DialogType::SecretChat);
send_get_dialog_query(dialog_id, Auto()); repair_server_unread_count(dialog_id, d->server_unread_count);
} }
update_dialog_pos(d, false, "fix_new_dialog"); update_dialog_pos(d, false, "fix_new_dialog");

View File

@ -2088,6 +2088,8 @@ class MessagesManager : public Actor {
static int32 calc_new_unread_count(Dialog *d, MessageId max_message_id, MessageType type, int32 hint_unread_count); static int32 calc_new_unread_count(Dialog *d, MessageId max_message_id, MessageType type, int32 hint_unread_count);
void repair_server_unread_count(DialogId dialog_id, int32 unread_count);
void read_history_inbox(DialogId dialog_id, MessageId max_message_id, int32 unread_count, const char *source); void read_history_inbox(DialogId dialog_id, MessageId max_message_id, int32 unread_count, const char *source);
void read_history_outbox(DialogId dialog_id, MessageId max_message_id, int32 read_date = -1); void read_history_outbox(DialogId dialog_id, MessageId max_message_id, int32 read_date = -1);

View File

@ -404,7 +404,7 @@ class CliClient final : public Actor {
static int64 as_message_id(Slice str) { static int64 as_message_id(Slice str) {
str = trim(str); str = trim(str);
if (!str.empty() && str.back() == 's') { if (!str.empty() && str.back() == 's') {
return to_integer<int32>(str) << 20; return to_integer<int64>(str) << 20;
} }
return to_integer<int64>(str); return to_integer<int64>(str);
} }