mirror of https://github.com/procxx/kepka.git
First version of reading-while-scrolling.
This commit is contained in:
parent
db2aa7000a
commit
70408f0e22
|
@ -6009,13 +6009,16 @@ void ApiWrap::readServerHistory(not_null<History*> history) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ApiWrap::readServerHistoryForce(not_null<History*> history) {
|
void ApiWrap::readServerHistoryForce(
|
||||||
|
not_null<History*> history,
|
||||||
|
MsgId upTo) {
|
||||||
const auto peer = history->peer;
|
const auto peer = history->peer;
|
||||||
const auto upTo = history->readInbox();
|
|
||||||
if (!upTo) {
|
if (!upTo) {
|
||||||
return;
|
upTo = history->readInbox();
|
||||||
|
if (!upTo) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (const auto channel = peer->asChannel()) {
|
if (const auto channel = peer->asChannel()) {
|
||||||
if (!channel->amIn()) {
|
if (!channel->amIn()) {
|
||||||
return; // no read request for channels that I didn't join
|
return; // no read request for channels that I didn't join
|
||||||
|
@ -6099,7 +6102,7 @@ void ApiWrap::sendReadRequest(not_null<PeerData*> peer, MsgId upTo) {
|
||||||
sendReadRequest(peer, *next);
|
sendReadRequest(peer, *next);
|
||||||
} else if (const auto history
|
} else if (const auto history
|
||||||
= _session->data().historyLoaded(peer)) {
|
= _session->data().historyLoaded(peer)) {
|
||||||
if (!history->unreadCountKnown()) {
|
if (history->unreadCountRefreshNeeded()) {
|
||||||
requestDialogEntry(history);
|
requestDialogEntry(history);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -390,7 +390,7 @@ public:
|
||||||
const SendAction &action);
|
const SendAction &action);
|
||||||
void shareContact(not_null<UserData*> user, const SendAction &action);
|
void shareContact(not_null<UserData*> user, const SendAction &action);
|
||||||
void readServerHistory(not_null<History*> history);
|
void readServerHistory(not_null<History*> history);
|
||||||
void readServerHistoryForce(not_null<History*> history);
|
void readServerHistoryForce(not_null<History*> history, MsgId upTo = 0);
|
||||||
//void readFeed( // #feed
|
//void readFeed( // #feed
|
||||||
// not_null<Data::Feed*> feed,
|
// not_null<Data::Feed*> feed,
|
||||||
// Data::MessagePosition position);
|
// Data::MessagePosition position);
|
||||||
|
|
|
@ -1592,11 +1592,12 @@ void History::calculateFirstUnreadMessage() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void History::readClientSideMessages() {
|
void History::readClientSideMessages() {
|
||||||
for (const auto &block : blocks) {
|
auto unread = unreadCount();
|
||||||
for (const auto &view : block->messages) {
|
for (const auto item : _localMessages) {
|
||||||
const auto item = view->data();
|
if (!item->out() && item->unread()) {
|
||||||
if (!item->out()) {
|
item->markClientSideAsRead();
|
||||||
item->markClientSideAsRead();
|
if (unread > 0) {
|
||||||
|
setUnreadCount(--unread);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1604,16 +1605,117 @@ void History::readClientSideMessages() {
|
||||||
|
|
||||||
MsgId History::readInbox() {
|
MsgId History::readInbox() {
|
||||||
const auto upTo = msgIdForRead();
|
const auto upTo = msgIdForRead();
|
||||||
|
readClientSideMessages();
|
||||||
if (unreadCountKnown()) {
|
if (unreadCountKnown()) {
|
||||||
setUnreadCount(0);
|
setUnreadCount(0);
|
||||||
}
|
}
|
||||||
readClientSideMessages();
|
|
||||||
if (upTo) {
|
if (upTo) {
|
||||||
inboxRead(upTo);
|
inboxRead(upTo);
|
||||||
}
|
}
|
||||||
return upTo;
|
return upTo;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void History::readInboxTill(not_null<HistoryItem*> item) {
|
||||||
|
if (!IsServerMsgId(item->id)) {
|
||||||
|
auto view = item->mainView();
|
||||||
|
if (!view) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto block = view->block();
|
||||||
|
auto blockIndex = block->indexInHistory();
|
||||||
|
auto itemIndex = view->indexInBlock();
|
||||||
|
while (blockIndex > 0 || itemIndex > 0) {
|
||||||
|
if (itemIndex > 0) {
|
||||||
|
view = block->messages[--itemIndex].get();
|
||||||
|
} else {
|
||||||
|
while (blockIndex > 0) {
|
||||||
|
block = blocks[--blockIndex].get();
|
||||||
|
itemIndex = block->messages.size();
|
||||||
|
if (itemIndex > 0) {
|
||||||
|
view = block->messages[--itemIndex].get();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
item = view->data();
|
||||||
|
if (IsServerMsgId(item->id)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!IsServerMsgId(item->id)) {
|
||||||
|
LOG(("App Error: "
|
||||||
|
"Can't read history till unknown local message."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
readClientSideMessages();
|
||||||
|
if (unreadMark()) {
|
||||||
|
session().api().changeDialogUnreadMark(this, false);
|
||||||
|
}
|
||||||
|
if (_inboxReadTillLocal >= item->id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_inboxReadTillLocal = item->id;
|
||||||
|
const auto stillUnread = countStillUnreadLocal();
|
||||||
|
if (!stillUnread) {
|
||||||
|
session().api().readServerHistoryForce(this, _inboxReadTillLocal);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setInboxReadTill(_inboxReadTillLocal);
|
||||||
|
if (stillUnread && _unreadCount && *stillUnread == *_unreadCount) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setUnreadCount(*stillUnread);
|
||||||
|
session().api().readServerHistoryForce(this, _inboxReadTillLocal);
|
||||||
|
updateChatListEntry();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool History::unreadCountRefreshNeeded() const {
|
||||||
|
return !unreadCountKnown()
|
||||||
|
|| ((_inboxReadTillLocal + 1) > _inboxReadBefore.value_or(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<int> History::countStillUnreadLocal() const {
|
||||||
|
if (isEmpty()) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
const auto till = _inboxReadTillLocal;
|
||||||
|
if (_inboxReadBefore) {
|
||||||
|
const auto before = *_inboxReadBefore;
|
||||||
|
if (minMsgId() <= before && maxMsgId() >= till) {
|
||||||
|
auto result = 0;
|
||||||
|
for (const auto &block : blocks) {
|
||||||
|
for (const auto &message : block->messages) {
|
||||||
|
const auto item = message->data();
|
||||||
|
if (item->out() || !IsServerMsgId(item->id)) {
|
||||||
|
continue;
|
||||||
|
} else if (item->id > till) {
|
||||||
|
break;
|
||||||
|
} else if (item->id >= before) {
|
||||||
|
++result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (_unreadCount) {
|
||||||
|
return std::max(*_unreadCount - result, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!loadedAtBottom() || minMsgId() > till) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
auto result = 0;
|
||||||
|
for (const auto &block : blocks) {
|
||||||
|
for (const auto &message : block->messages) {
|
||||||
|
const auto item = message->data();
|
||||||
|
if (!item->out() && IsServerMsgId(item->id) && item->id > till) {
|
||||||
|
++result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
void History::applyInboxReadUpdate(
|
void History::applyInboxReadUpdate(
|
||||||
FolderId folderId,
|
FolderId folderId,
|
||||||
MsgId upTo,
|
MsgId upTo,
|
||||||
|
@ -1625,10 +1727,12 @@ void History::applyInboxReadUpdate(
|
||||||
session().api().requestDialogEntry(this);
|
session().api().requestDialogEntry(this);
|
||||||
session().api().requestDialogEntry(folder);
|
session().api().requestDialogEntry(folder);
|
||||||
}
|
}
|
||||||
if (!peer->isChannel() || peer->asChannel()->pts() == channelPts) {
|
if (_inboxReadTillLocal <= upTo) {
|
||||||
inboxRead(upTo, stillUnread);
|
if (!peer->isChannel() || peer->asChannel()->pts() == channelPts) {
|
||||||
} else {
|
inboxRead(upTo, stillUnread);
|
||||||
inboxRead(upTo);
|
} else {
|
||||||
|
inboxRead(upTo);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1645,9 +1749,9 @@ void History::inboxRead(MsgId upTo, std::optional<int> stillUnread) {
|
||||||
}
|
}
|
||||||
setInboxReadTill(upTo);
|
setInboxReadTill(upTo);
|
||||||
updateChatListEntry();
|
updateChatListEntry();
|
||||||
if (peer->migrateTo()) {
|
if (const auto to = peer->migrateTo()) {
|
||||||
if (auto migrateTo = peer->owner().historyLoaded(peer->migrateTo()->id)) {
|
if (const auto migrated = peer->owner().historyLoaded(to->id)) {
|
||||||
migrateTo->updateChatListEntry();
|
migrated->updateChatListEntry();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2656,7 +2760,7 @@ void History::applyDialogFields(
|
||||||
} else {
|
} else {
|
||||||
clearFolder();
|
clearFolder();
|
||||||
}
|
}
|
||||||
if (!skipUnreadUpdate()) {
|
if (!skipUnreadUpdate() && maxInboxRead >= _inboxReadTillLocal) {
|
||||||
setUnreadCount(unreadCount);
|
setUnreadCount(unreadCount);
|
||||||
setInboxReadTill(maxInboxRead);
|
setInboxReadTill(maxInboxRead);
|
||||||
}
|
}
|
||||||
|
@ -2690,6 +2794,7 @@ void History::setInboxReadTill(MsgId upTo) {
|
||||||
} else {
|
} else {
|
||||||
_inboxReadBefore = upTo + 1;
|
_inboxReadBefore = upTo + 1;
|
||||||
}
|
}
|
||||||
|
accumulate_max(_inboxReadTillLocal, upTo);
|
||||||
}
|
}
|
||||||
|
|
||||||
void History::setOutboxReadTill(MsgId upTo) {
|
void History::setOutboxReadTill(MsgId upTo) {
|
||||||
|
|
|
@ -159,6 +159,7 @@ public:
|
||||||
[[nodiscard]] HistoryItem *latestSendingMessage() const;
|
[[nodiscard]] HistoryItem *latestSendingMessage() const;
|
||||||
|
|
||||||
MsgId readInbox();
|
MsgId readInbox();
|
||||||
|
void readInboxTill(not_null<HistoryItem*> item);
|
||||||
void applyInboxReadUpdate(
|
void applyInboxReadUpdate(
|
||||||
FolderId folderId,
|
FolderId folderId,
|
||||||
MsgId upTo,
|
MsgId upTo,
|
||||||
|
@ -174,6 +175,10 @@ public:
|
||||||
|
|
||||||
[[nodiscard]] int unreadCount() const;
|
[[nodiscard]] int unreadCount() const;
|
||||||
[[nodiscard]] bool unreadCountKnown() const;
|
[[nodiscard]] bool unreadCountKnown() const;
|
||||||
|
|
||||||
|
// Some old unread count is known, but we read history till some place.
|
||||||
|
[[nodiscard]] bool unreadCountRefreshNeeded() const;
|
||||||
|
|
||||||
void setUnreadCount(int newUnreadCount);
|
void setUnreadCount(int newUnreadCount);
|
||||||
void setUnreadMark(bool unread);
|
void setUnreadMark(bool unread);
|
||||||
[[nodiscard]] bool unreadMark() const;
|
[[nodiscard]] bool unreadMark() const;
|
||||||
|
@ -469,6 +474,7 @@ private:
|
||||||
void getNextFirstUnreadMessage();
|
void getNextFirstUnreadMessage();
|
||||||
bool nonEmptyCountMoreThan(int count) const;
|
bool nonEmptyCountMoreThan(int count) const;
|
||||||
std::optional<int> countUnread(MsgId upTo) const;
|
std::optional<int> countUnread(MsgId upTo) const;
|
||||||
|
std::optional<int> countStillUnreadLocal() const;
|
||||||
|
|
||||||
// Creates if necessary a new block for adding item.
|
// Creates if necessary a new block for adding item.
|
||||||
// Depending on isBuildingFrontBlock() gets front or back block.
|
// Depending on isBuildingFrontBlock() gets front or back block.
|
||||||
|
@ -497,6 +503,7 @@ private:
|
||||||
|
|
||||||
std::optional<MsgId> _inboxReadBefore;
|
std::optional<MsgId> _inboxReadBefore;
|
||||||
std::optional<MsgId> _outboxReadBefore;
|
std::optional<MsgId> _outboxReadBefore;
|
||||||
|
MsgId _inboxReadTillLocal = 0;
|
||||||
std::optional<int> _unreadCount;
|
std::optional<int> _unreadCount;
|
||||||
std::optional<int> _unreadMentionsCount;
|
std::optional<int> _unreadMentionsCount;
|
||||||
base::flat_set<MsgId> _unreadMentions;
|
base::flat_set<MsgId> _unreadMentions;
|
||||||
|
|
|
@ -31,6 +31,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "window/window_session_controller.h"
|
#include "window/window_session_controller.h"
|
||||||
#include "window/window_peer_menu.h"
|
#include "window/window_peer_menu.h"
|
||||||
#include "window/window_controller.h"
|
#include "window/window_controller.h"
|
||||||
|
#include "window/notifications_manager.h"
|
||||||
#include "boxes/confirm_box.h"
|
#include "boxes/confirm_box.h"
|
||||||
#include "boxes/report_box.h"
|
#include "boxes/report_box.h"
|
||||||
#include "boxes/sticker_set_box.h"
|
#include "boxes/sticker_set_box.h"
|
||||||
|
@ -655,13 +656,13 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
|
||||||
auto iItem = (_curHistory == _history ? _curItem : 0);
|
auto iItem = (_curHistory == _history ? _curItem : 0);
|
||||||
auto view = block->messages[iItem].get();
|
auto view = block->messages[iItem].get();
|
||||||
auto item = view->data();
|
auto item = view->data();
|
||||||
|
auto readTill = (HistoryItem*)nullptr;
|
||||||
auto hclip = clip.intersected(QRect(0, hdrawtop, width(), clip.top() + clip.height()));
|
auto hclip = clip.intersected(QRect(0, hdrawtop, width(), clip.top() + clip.height()));
|
||||||
auto y = htop + block->y() + view->y();
|
auto y = htop + block->y() + view->y();
|
||||||
p.save();
|
p.save();
|
||||||
p.translate(0, y);
|
p.translate(0, y);
|
||||||
while (y < drawToY) {
|
while (y < drawToY) {
|
||||||
auto h = view->height();
|
const auto h = view->height();
|
||||||
if (hclip.y() < y + h && hdrawtop < y + h) {
|
if (hclip.y() < y + h && hdrawtop < y + h) {
|
||||||
const auto selection = itemRenderSelection(
|
const auto selection = itemRenderSelection(
|
||||||
view,
|
view,
|
||||||
|
@ -669,12 +670,20 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
|
||||||
seltoy - htop);
|
seltoy - htop);
|
||||||
view->draw(p, hclip.translated(0, -y), selection, ms);
|
view->draw(p, hclip.translated(0, -y), selection, ms);
|
||||||
|
|
||||||
if (item->hasViews()) {
|
const auto middle = y + h / 2;
|
||||||
App::main()->scheduleViewIncrement(item);
|
const auto bottom = y + h;
|
||||||
|
if (_visibleAreaBottom >= bottom) {
|
||||||
|
readTill = view->data();
|
||||||
}
|
}
|
||||||
if (item->isUnreadMention() && !item->isUnreadMedia()) {
|
if (_visibleAreaBottom >= middle
|
||||||
readMentions.insert(item);
|
&& _visibleAreaTop <= middle) {
|
||||||
_widget->enqueueMessageHighlight(view);
|
if (item->hasViews()) {
|
||||||
|
App::main()->scheduleViewIncrement(item);
|
||||||
|
}
|
||||||
|
if (item->isUnreadMention() && !item->isUnreadMedia()) {
|
||||||
|
readMentions.insert(item);
|
||||||
|
_widget->enqueueMessageHighlight(view);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
p.translate(0, h);
|
p.translate(0, h);
|
||||||
|
@ -693,9 +702,13 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
|
||||||
item = view->data();
|
item = view->data();
|
||||||
}
|
}
|
||||||
p.restore();
|
p.restore();
|
||||||
|
|
||||||
|
if (readTill) {
|
||||||
|
_history->readInboxTill(readTill);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!readMentions.empty() && App::wnd()->doWeReadMentions()) {
|
if (!readMentions.empty() && _widget->doWeReadMentions()) {
|
||||||
session().api().markMediaRead(readMentions);
|
session().api().markMediaRead(readMentions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2013,6 +2026,42 @@ void HistoryInner::keyPressEvent(QKeyEvent *e) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HistoryInner::checkHistoryActivation() {
|
||||||
|
if (!_widget->doWeReadServerHistory()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
adjustCurrent(_visibleAreaBottom);
|
||||||
|
if (_history->loadedAtBottom() && _visibleAreaBottom >= height()) {
|
||||||
|
// Clear possible scheduled messages notifications.
|
||||||
|
session().notifications().clearFromHistory(_history);
|
||||||
|
}
|
||||||
|
if (_curHistory != _history || _history->isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto block = _history->blocks[_curBlock].get();
|
||||||
|
auto view = block->messages[_curItem].get();
|
||||||
|
while (_curBlock > 0 || _curItem > 0) {
|
||||||
|
const auto top = itemTop(view);
|
||||||
|
const auto bottom = itemTop(view) + view->height();
|
||||||
|
if (_visibleAreaBottom >= bottom) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (_curItem > 0) {
|
||||||
|
view = block->messages[--_curItem].get();
|
||||||
|
} else {
|
||||||
|
while (_curBlock > 0) {
|
||||||
|
block = _history->blocks[--_curBlock].get();
|
||||||
|
_curItem = block->messages.size();
|
||||||
|
if (_curItem > 0) {
|
||||||
|
view = block->messages[--_curItem].get();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_history->readInboxTill(view->data());
|
||||||
|
}
|
||||||
|
|
||||||
void HistoryInner::recountHistoryGeometry() {
|
void HistoryInner::recountHistoryGeometry() {
|
||||||
_contentWidth = _scroll->width();
|
_contentWidth = _scroll->width();
|
||||||
|
|
||||||
|
@ -2178,6 +2227,7 @@ void HistoryInner::visibleAreaUpdated(int top, int bottom) {
|
||||||
const auto from = _visibleAreaTop - pages * visibleAreaHeight;
|
const auto from = _visibleAreaTop - pages * visibleAreaHeight;
|
||||||
const auto till = _visibleAreaBottom + pages * visibleAreaHeight;
|
const auto till = _visibleAreaBottom + pages * visibleAreaHeight;
|
||||||
session().data().unloadHeavyViewParts(ElementDelegate(), from, till);
|
session().data().unloadHeavyViewParts(ElementDelegate(), from, till);
|
||||||
|
checkHistoryActivation();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryInner::displayScrollDate() const {
|
bool HistoryInner::displayScrollDate() const {
|
||||||
|
@ -2326,7 +2376,8 @@ void HistoryInner::adjustCurrent(int32 y) const {
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryInner::adjustCurrent(int32 y, History *history) const {
|
void HistoryInner::adjustCurrent(int32 y, History *history) const {
|
||||||
Assert(!history->isEmpty());
|
Expects(!history->isEmpty());
|
||||||
|
|
||||||
_curHistory = history;
|
_curHistory = history;
|
||||||
if (_curBlock >= history->blocks.size()) {
|
if (_curBlock >= history->blocks.size()) {
|
||||||
_curBlock = history->blocks.size() - 1;
|
_curBlock = history->blocks.size() - 1;
|
||||||
|
|
|
@ -62,6 +62,7 @@ public:
|
||||||
|
|
||||||
void touchScrollUpdated(const QPoint &screenPos);
|
void touchScrollUpdated(const QPoint &screenPos);
|
||||||
|
|
||||||
|
void checkHistoryActivation();
|
||||||
void recountHistoryGeometry();
|
void recountHistoryGeometry();
|
||||||
void updateSize();
|
void updateSize();
|
||||||
|
|
||||||
|
|
|
@ -2265,16 +2265,18 @@ void HistoryWidget::unreadMessageAdded(not_null<HistoryItem*> item) {
|
||||||
// - on second we get wrong doWeReadServerHistory() and read both.
|
// - on second we get wrong doWeReadServerHistory() and read both.
|
||||||
session().data().sendHistoryChangeNotifications();
|
session().data().sendHistoryChangeNotifications();
|
||||||
|
|
||||||
if (_scroll->scrollTop() + 1 > _scroll->scrollTopMax()) {
|
const auto atBottom = (_scroll->scrollTop() >= _scroll->scrollTopMax());
|
||||||
destroyUnreadBar();
|
if (!atBottom) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
if (!App::wnd()->doWeReadServerHistory()) {
|
destroyUnreadBar();
|
||||||
|
if (!doWeReadServerHistory()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (item->isUnreadMention() && !item->isUnreadMedia()) {
|
if (item->isUnreadMention() && !item->isUnreadMedia()) {
|
||||||
session().api().markMediaRead(item);
|
session().api().markMediaRead(item);
|
||||||
}
|
}
|
||||||
session().api().readServerHistoryForce(_history);
|
_history->readInboxTill(item);
|
||||||
|
|
||||||
// Also clear possible scheduled messages notifications.
|
// Also clear possible scheduled messages notifications.
|
||||||
session().notifications().clearFromHistory(_history);
|
session().notifications().clearFromHistory(_history);
|
||||||
|
@ -2403,7 +2405,9 @@ void HistoryWidget::messagesReceived(PeerData *peer, const MTPmessages_Messages
|
||||||
addMessagesToBack(peer, *histList);
|
addMessagesToBack(peer, *histList);
|
||||||
_preloadDownRequest = 0;
|
_preloadDownRequest = 0;
|
||||||
preloadHistoryIfNeeded();
|
preloadHistoryIfNeeded();
|
||||||
if (_history->loadedAtBottom() && App::wnd()) App::wnd()->checkHistoryActivation();
|
if (_history->loadedAtBottom()) {
|
||||||
|
App::wnd()->checkHistoryActivation();
|
||||||
|
}
|
||||||
} else if (_firstLoadRequest == requestId) {
|
} else if (_firstLoadRequest == requestId) {
|
||||||
if (toMigrated) {
|
if (toMigrated) {
|
||||||
_history->clear(History::ClearType::Unload);
|
_history->clear(History::ClearType::Unload);
|
||||||
|
@ -2472,30 +2476,21 @@ void HistoryWidget::windowShown() {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryWidget::doWeReadServerHistory() const {
|
bool HistoryWidget::doWeReadServerHistory() const {
|
||||||
if (!_history || !_list) return true;
|
return doWeReadMentions() && !session().supportMode();
|
||||||
if (_firstLoadRequest || _a_show.animating()) return false;
|
|
||||||
if (_history->loadedAtBottom()) {
|
|
||||||
int scrollTop = _scroll->scrollTop();
|
|
||||||
if (scrollTop + 1 > _scroll->scrollTopMax()) return true;
|
|
||||||
|
|
||||||
if (const auto unread = firstUnreadMessage()) {
|
|
||||||
const auto scrollBottom = scrollTop + _scroll->height();
|
|
||||||
if (scrollBottom > _list->itemTop(unread)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (_history->hasNotFreezedUnreadBar()
|
|
||||||
|| (_migrated && _migrated->hasNotFreezedUnreadBar())) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryWidget::doWeReadMentions() const {
|
bool HistoryWidget::doWeReadMentions() const {
|
||||||
if (!_history || !_list) return true;
|
return _history
|
||||||
if (_firstLoadRequest || _a_show.animating()) return false;
|
&& _list
|
||||||
return true;
|
&& !_firstLoadRequest
|
||||||
|
&& !_a_show.animating()
|
||||||
|
&& App::wnd()->doWeMarkAsRead();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryWidget::checkHistoryActivation() {
|
||||||
|
if (_list) {
|
||||||
|
_list->checkHistoryActivation();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryWidget::firstLoadMessages() {
|
void HistoryWidget::firstLoadMessages() {
|
||||||
|
@ -2714,24 +2709,6 @@ void HistoryWidget::visibleAreaUpdated() {
|
||||||
const auto scrollBottom = scrollTop + _scroll->height();
|
const auto scrollBottom = scrollTop + _scroll->height();
|
||||||
_list->visibleAreaUpdated(scrollTop, scrollBottom);
|
_list->visibleAreaUpdated(scrollTop, scrollBottom);
|
||||||
controller()->floatPlayerAreaUpdated().notify(true);
|
controller()->floatPlayerAreaUpdated().notify(true);
|
||||||
|
|
||||||
const auto atBottom = (scrollTop >= _scroll->scrollTopMax());
|
|
||||||
if (_history->loadedAtBottom()
|
|
||||||
&& atBottom
|
|
||||||
&& App::wnd()->doWeReadServerHistory()) {
|
|
||||||
// Clear possible scheduled messages notifications.
|
|
||||||
session().api().readServerHistory(_history);
|
|
||||||
session().notifications().clearFromHistory(_history);
|
|
||||||
} else if (_history->loadedAtBottom()
|
|
||||||
&& (_history->unreadCount() > 0
|
|
||||||
|| (_migrated && _migrated->unreadCount() > 0))) {
|
|
||||||
const auto unread = firstUnreadMessage();
|
|
||||||
const auto unreadVisible = unread
|
|
||||||
&& (scrollBottom > _list->itemTop(unread));
|
|
||||||
if (unreadVisible && App::wnd()->doWeReadServerHistory()) {
|
|
||||||
session().api().readServerHistory(_history);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2820,9 +2797,8 @@ void HistoryWidget::historyDownClicked() {
|
||||||
} else if (_replyReturn && _replyReturn->history() == _migrated) {
|
} else if (_replyReturn && _replyReturn->history() == _migrated) {
|
||||||
showHistory(_peer->id, -_replyReturn->id);
|
showHistory(_peer->id, -_replyReturn->id);
|
||||||
} else if (_peer) {
|
} else if (_peer) {
|
||||||
showHistory(
|
showHistory(_peer->id, ShowAtTheEndMsgId); // #TODO reading
|
||||||
_peer->id,
|
// session().supportMode() ? ShowAtTheEndMsgId : ShowAtUnreadMsgId);
|
||||||
session().supportMode() ? ShowAtTheEndMsgId : ShowAtUnreadMsgId);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3178,10 +3154,8 @@ void HistoryWidget::doneShow() {
|
||||||
handlePendingHistoryUpdate();
|
handlePendingHistoryUpdate();
|
||||||
}
|
}
|
||||||
preloadHistoryIfNeeded();
|
preloadHistoryIfNeeded();
|
||||||
if (App::wnd()) {
|
App::wnd()->checkHistoryActivation();
|
||||||
App::wnd()->checkHistoryActivation();
|
App::wnd()->setInnerFocus();
|
||||||
App::wnd()->setInnerFocus();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryWidget::finishAnimating() {
|
void HistoryWidget::finishAnimating() {
|
||||||
|
|
|
@ -109,9 +109,10 @@ public:
|
||||||
void historyLoaded();
|
void historyLoaded();
|
||||||
|
|
||||||
void windowShown();
|
void windowShown();
|
||||||
bool doWeReadServerHistory() const;
|
[[nodiscard]] bool doWeReadServerHistory() const;
|
||||||
bool doWeReadMentions() const;
|
[[nodiscard]] bool doWeReadMentions() const;
|
||||||
bool skipItemRepaint();
|
bool skipItemRepaint();
|
||||||
|
void checkHistoryActivation();
|
||||||
|
|
||||||
void leaveToChildEvent(QEvent *e, QWidget *child) override;
|
void leaveToChildEvent(QEvent *e, QWidget *child) override;
|
||||||
void dragEnterEvent(QDragEnterEvent *e) override;
|
void dragEnterEvent(QDragEnterEvent *e) override;
|
||||||
|
|
|
@ -2217,10 +2217,8 @@ void MainWidget::dialogsToUp() {
|
||||||
_dialogs->jumpToTop();
|
_dialogs->jumpToTop();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWidget::markActiveHistoryAsRead() {
|
void MainWidget::checkHistoryActivation() {
|
||||||
if (const auto activeHistory = _history->history()) {
|
_history->checkHistoryActivation();
|
||||||
session().api().readServerHistory(activeHistory);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWidget::showAnimated(const QPixmap &bgAnimCache, bool back) {
|
void MainWidget::showAnimated(const QPixmap &bgAnimCache, bool back) {
|
||||||
|
@ -3519,15 +3517,8 @@ bool MainWidget::isActive() const {
|
||||||
return !_isIdle && isVisible() && !_a_show.animating();
|
return !_isIdle && isVisible() && !_a_show.animating();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MainWidget::doWeReadServerHistory() const {
|
bool MainWidget::doWeMarkAsRead() const {
|
||||||
return isActive()
|
return isActive() && !_mainSection;
|
||||||
&& !session().supportMode()
|
|
||||||
&& !_mainSection
|
|
||||||
&& _history->doWeReadServerHistory();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool MainWidget::doWeReadMentions() const {
|
|
||||||
return isActive() && !_mainSection && _history->doWeReadMentions();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MainWidget::lastWasOnline() const {
|
bool MainWidget::lastWasOnline() const {
|
||||||
|
|
|
@ -148,7 +148,7 @@ public:
|
||||||
bool deleteChannelFailed(const RPCError &error);
|
bool deleteChannelFailed(const RPCError &error);
|
||||||
void historyToDown(History *hist);
|
void historyToDown(History *hist);
|
||||||
void dialogsToUp();
|
void dialogsToUp();
|
||||||
void markActiveHistoryAsRead();
|
void checkHistoryActivation();
|
||||||
|
|
||||||
PeerData *peer();
|
PeerData *peer();
|
||||||
|
|
||||||
|
@ -173,8 +173,7 @@ public:
|
||||||
void updateOnlineDisplayIn(int32 msecs);
|
void updateOnlineDisplayIn(int32 msecs);
|
||||||
|
|
||||||
bool isActive() const;
|
bool isActive() const;
|
||||||
bool doWeReadServerHistory() const;
|
[[nodiscard]] bool doWeMarkAsRead() const;
|
||||||
bool doWeReadMentions() const;
|
|
||||||
bool lastWasOnline() const;
|
bool lastWasOnline() const;
|
||||||
crl::time lastSetOnline() const;
|
crl::time lastSetOnline() const;
|
||||||
|
|
||||||
|
|
|
@ -503,23 +503,17 @@ void MainWindow::themeUpdated(const Window::Theme::BackgroundUpdate &data) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MainWindow::doWeReadServerHistory() {
|
bool MainWindow::doWeMarkAsRead() {
|
||||||
|
if (!_main || Ui::isLayerShown()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
updateIsActive(0);
|
updateIsActive(0);
|
||||||
return isActive()
|
return isActive();
|
||||||
&& !Ui::isLayerShown()
|
|
||||||
&& (_main ? _main->doWeReadServerHistory() : false);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool MainWindow::doWeReadMentions() {
|
|
||||||
updateIsActive(0);
|
|
||||||
return isActive()
|
|
||||||
&& !Ui::isLayerShown()
|
|
||||||
&& (_main ? _main->doWeReadMentions() : false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::checkHistoryActivation() {
|
void MainWindow::checkHistoryActivation() {
|
||||||
if (doWeReadServerHistory()) {
|
if (_main) {
|
||||||
_main->markActiveHistoryAsRead();
|
_main->checkHistoryActivation();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -62,8 +62,7 @@ public:
|
||||||
|
|
||||||
MainWidget *mainWidget();
|
MainWidget *mainWidget();
|
||||||
|
|
||||||
bool doWeReadServerHistory();
|
[[nodiscard]] bool doWeMarkAsRead();
|
||||||
bool doWeReadMentions();
|
|
||||||
|
|
||||||
void activate();
|
void activate();
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue