Improve working with unread bar.

This commit is contained in:
John Preston 2020-02-20 19:25:31 +04:00
parent ee3e9af63a
commit 49c4d35afa
3 changed files with 89 additions and 116 deletions

View File

@ -1590,22 +1590,22 @@ void History::addToSharedMedia(
} }
void History::calculateFirstUnreadMessage() { void History::calculateFirstUnreadMessage() {
if (_firstUnreadView || !_inboxReadBefore) { if (!_inboxReadBefore) {
return; return;
} }
for (auto i = blocks.cend(); i != blocks.cbegin();) { _firstUnreadView = nullptr;
--i; if (!unreadCount()) {
const auto &messages = (*i)->messages; return;
for (auto j = messages.cend(); j != messages.cbegin();) { }
--j; for (const auto &block : ranges::view::reverse(blocks)) {
const auto view = j->get(); for (const auto &message : ranges::view::reverse(block->messages)) {
const auto item = view->data(); const auto item = message->data();
if (!IsServerMsgId(item->id)) { if (!IsServerMsgId(item->id)) {
continue; continue;
} else if (!item->out() || !_firstUnreadView) { } else if (!item->out()) {
if (item->id >= *_inboxReadBefore) { if (item->id >= *_inboxReadBefore) {
_firstUnreadView = view; _firstUnreadView = message.get();
} else { } else {
return; return;
} }
@ -1798,10 +1798,8 @@ void History::setUnreadCount(int newUnreadCount) {
if (const auto last = msgIdForRead()) { if (const auto last = msgIdForRead()) {
setInboxReadTill(last); setInboxReadTill(last);
} }
} else { } else if (!_firstUnreadView && !_unreadBarView && loadedAtBottom()) {
if (!_firstUnreadView && !_unreadBarView && loadedAtBottom()) { calculateFirstUnreadMessage();
calculateFirstUnreadMessage();
}
} }
Notify::peerUpdatedDelayed( Notify::peerUpdatedDelayed(
peer, peer,

View File

@ -1551,10 +1551,7 @@ void HistoryWidget::fastShowAtEnd(not_null<History*> history) {
} }
clearAllLoadRequests(); clearAllLoadRequests();
setMsgId(ShowAtUnreadMsgId); setMsgId(ShowAtUnreadMsgId);
_historyInited = false;
if (_history->isReadyFor(_showAtMsgId)) { if (_history->isReadyFor(_showAtMsgId)) {
historyLoaded(); historyLoaded();
} else { } else {
@ -1637,7 +1634,7 @@ void HistoryWidget::showHistory(
if (_peer->id == peerId && !reload) { if (_peer->id == peerId && !reload) {
updateForwarding(); updateForwarding();
bool canShowNow = _history->isReadyFor(showAtMsgId); const auto canShowNow = _history->isReadyFor(showAtMsgId);
if (!canShowNow) { if (!canShowNow) {
delayedShowAt(showAtMsgId); delayedShowAt(showAtMsgId);
} else { } else {
@ -1659,10 +1656,8 @@ void HistoryWidget::showHistory(
setMsgId(showAtMsgId); setMsgId(showAtMsgId);
if (_historyInited) { if (_historyInited) {
countHistoryShowFrom(); const auto item = getItemFromHistoryOrMigrated(
destroyUnreadBar(); _showAtMsgId);
auto item = getItemFromHistoryOrMigrated(_showAtMsgId);
animatedScrollToY(countInitialScrollTop(), item); animatedScrollToY(countInitialScrollTop(), item);
} else { } else {
historyLoaded(); historyLoaded();
@ -1712,7 +1707,7 @@ void HistoryWidget::showHistory(
_history->showAtMsgId = _showAtMsgId; _history->showAtMsgId = _showAtMsgId;
destroyUnreadBar(); destroyUnreadBarOnClose();
destroyPinnedBar(); destroyPinnedBar();
_membersDropdown.destroy(); _membersDropdown.destroy();
_scrollToAnimation.stop(); _scrollToAnimation.stop();
@ -1822,7 +1817,9 @@ void HistoryWidget::showHistory(
_updateHistoryItems.stop(); _updateHistoryItems.stop();
pinnedMsgVisibilityUpdated(); pinnedMsgVisibilityUpdated();
if (_history->scrollTopItem || (_migrated && _migrated->scrollTopItem) || _history->isReadyFor(_showAtMsgId)) { if (_history->scrollTopItem
|| (_migrated && _migrated->scrollTopItem)
|| _history->isReadyFor(_showAtMsgId)) {
historyLoaded(); historyLoaded();
} else { } else {
firstLoadMessages(); firstLoadMessages();
@ -2236,8 +2233,22 @@ void HistoryWidget::destroyUnreadBar() {
if (_migrated) _migrated->destroyUnreadBar(); if (_migrated) _migrated->destroyUnreadBar();
} }
void HistoryWidget::destroyUnreadBarOnClose() {
if (!_history || !_historyInited) {
return;
} else if (_scroll->scrollTop() == _scroll->scrollTopMax()) {
destroyUnreadBar();
return;
}
const auto top = unreadBarTop();
if (top && *top < _scroll->scrollTop()) {
destroyUnreadBar();
return;
}
}
void HistoryWidget::unreadMessageAdded(not_null<HistoryItem*> item) { void HistoryWidget::unreadMessageAdded(not_null<HistoryItem*> item) {
if (_history != item->history()) { if (_history != item->history() || !_historyInited) {
return; return;
} }
@ -2442,15 +2453,12 @@ void HistoryWidget::messagesReceived(PeerData *peer, const MTPmessages_Messages
} }
setMsgId(_delayedShowAtMsgId); setMsgId(_delayedShowAtMsgId);
_historyInited = false;
historyLoaded(); historyLoaded();
} }
} }
void HistoryWidget::historyLoaded() { void HistoryWidget::historyLoaded() {
countHistoryShowFrom(); _historyInited = false;
destroyUnreadBar();
doneShow(); doneShow();
} }
@ -2478,7 +2486,9 @@ void HistoryWidget::checkHistoryActivation() {
} }
void HistoryWidget::firstLoadMessages() { void HistoryWidget::firstLoadMessages() {
if (!_history || _firstLoadRequest) return; if (!_history || _firstLoadRequest) {
return;
}
auto from = _peer; auto from = _peer;
auto offsetId = 0; auto offsetId = 0;
@ -2535,13 +2545,18 @@ void HistoryWidget::firstLoadMessages() {
} }
void HistoryWidget::loadMessages() { void HistoryWidget::loadMessages() {
if (!_history || _preloadRequest) return; if (!_history || _preloadRequest) {
return;
}
if (_history->isEmpty() && _migrated && _migrated->isEmpty()) { if (_history->isEmpty() && _migrated && _migrated->isEmpty()) {
return firstLoadMessages(); return firstLoadMessages();
} }
auto loadMigrated = _migrated && (_history->isEmpty() || _history->loadedAtTop() || (!_migrated->isEmpty() && !_migrated->loadedAtBottom())); auto loadMigrated = _migrated
&& (_history->isEmpty()
|| _history->loadedAtTop()
|| (!_migrated->isEmpty() && !_migrated->loadedAtBottom()));
auto from = loadMigrated ? _migrated : _history; auto from = loadMigrated ? _migrated : _history;
if (from->loadedAtTop()) { if (from->loadedAtTop()) {
return; return;
@ -2572,7 +2587,9 @@ void HistoryWidget::loadMessages() {
} }
void HistoryWidget::loadMessagesDown() { void HistoryWidget::loadMessagesDown() {
if (!_history || _preloadDownRequest) return; if (!_history || _preloadDownRequest) {
return;
}
if (_history->isEmpty() && _migrated && _migrated->isEmpty()) { if (_history->isEmpty() && _migrated && _migrated->isEmpty()) {
return firstLoadMessages(); return firstLoadMessages();
@ -4949,9 +4966,11 @@ int HistoryWidget::countInitialScrollTop() {
auto result = ScrollMax; auto result = ScrollMax;
if (_history->scrollTopItem || (_migrated && _migrated->scrollTopItem)) { if (_history->scrollTopItem || (_migrated && _migrated->scrollTopItem)) {
result = _list->historyScrollTop(); result = _list->historyScrollTop();
} else if (_showAtMsgId && (_showAtMsgId > 0 || -_showAtMsgId < ServerMaxMsgId)) { } else if (_showAtMsgId
auto item = getItemFromHistoryOrMigrated(_showAtMsgId); && (IsServerMsgId(_showAtMsgId)
auto itemTop = _list->itemTop(item); || IsServerMsgId(-_showAtMsgId))) {
const auto item = getItemFromHistoryOrMigrated(_showAtMsgId);
const auto itemTop = _list->itemTop(item);
if (itemTop < 0) { if (itemTop < 0) {
setMsgId(0); setMsgId(0);
return countInitialScrollTop(); return countInitialScrollTop();
@ -4971,8 +4990,14 @@ int HistoryWidget::countInitialScrollTop() {
} }
int HistoryWidget::countAutomaticScrollTop() { int HistoryWidget::countAutomaticScrollTop() {
Expects(_history != nullptr);
Expects(_list != nullptr);
auto result = ScrollMax; auto result = ScrollMax;
if (const auto unread = firstUnreadMessage()) { if (!_historyInited) {
_history->calculateFirstUnreadMessage();
}
if (const auto unread = _history->firstUnreadMessage()) {
result = _list->itemTop(unread); result = _list->itemTop(unread);
const auto possibleUnreadBarTop = _scroll->scrollTopMax() const auto possibleUnreadBarTop = _scroll->scrollTopMax()
+ HistoryView::UnreadBar::height() + HistoryView::UnreadBar::height()
@ -5037,10 +5062,10 @@ void HistoryWidget::updateHistoryGeometry(
if (newScrollHeight <= 0) { if (newScrollHeight <= 0) {
return; return;
} }
auto wasScrollTop = _scroll->scrollTop(); const auto wasScrollTop = _scroll->scrollTop();
auto wasScrollTopMax = _scroll->scrollTopMax(); const auto wasAtBottom = (wasScrollTop == _scroll->scrollTopMax());
auto wasAtBottom = wasScrollTop + 1 > wasScrollTopMax; const auto needResize = (_scroll->width() != width())
auto needResize = (_scroll->width() != width()) || (_scroll->height() != newScrollHeight); || (_scroll->height() != newScrollHeight);
if (needResize) { if (needResize) {
_scroll->resize(width(), newScrollHeight); _scroll->resize(width(), newScrollHeight);
// on initial updateListSize we didn't put the _scroll->scrollTop correctly yet // on initial updateListSize we didn't put the _scroll->scrollTop correctly yet
@ -5069,52 +5094,30 @@ void HistoryWidget::updateHistoryGeometry(
updateListSize(); updateListSize();
_updateHistoryGeometryRequired = false; _updateHistoryGeometryRequired = false;
if ((!initial && !wasAtBottom) auto newScrollTop = 0;
|| (loadedDown
&& (!_history->firstUnreadMessage()
|| _history->unreadBar()
|| _history->loadedAtBottom())
&& (!_migrated
|| !_migrated->firstUnreadMessage()
|| _migrated->unreadBar()
|| _history->loadedAtBottom()))) {
const auto historyScrollTop = _list->historyScrollTop();
if (!wasAtBottom && historyScrollTop == ScrollMax) {
// History scroll top was not inited yet.
// If we're showing locally unread messages, we get here
// from destroyUnreadBar() before we have time to scroll
// to good initial position, like top of an unread bar.
return;
}
auto toY = qMin(_list->historyScrollTop(), _scroll->scrollTopMax());
if (change.type == ScrollChangeAdd) {
toY += change.value;
} else if (change.type == ScrollChangeNoJumpToBottom) {
toY = wasScrollTop;
} else if (_addToScroll) {
toY += _addToScroll;
_addToScroll = 0;
}
toY = snap(toY, 0, _scroll->scrollTopMax());
if (_scroll->scrollTop() == toY) {
visibleAreaUpdated();
} else {
synteticScrollToY(toY);
}
return;
}
if (initial) { if (initial) {
newScrollTop = countInitialScrollTop();
_historyInited = true; _historyInited = true;
_scrollToAnimation.stop(); _scrollToAnimation.stop();
} else if (wasAtBottom && !loadedDown) {
newScrollTop = countAutomaticScrollTop();
} else {
newScrollTop = std::min(
_list->historyScrollTop(),
_scroll->scrollTopMax());
if (change.type == ScrollChangeAdd) {
newScrollTop += change.value;
} else if (change.type == ScrollChangeNoJumpToBottom) {
newScrollTop = wasScrollTop;
} else if (const auto add = base::take(_addToScroll)) {
newScrollTop += add;
}
} }
const auto newScrollTop = initial const auto toY = std::clamp(newScrollTop, 0, _scroll->scrollTopMax());
? countInitialScrollTop() if (_scroll->scrollTop() == toY) {
: countAutomaticScrollTop();
if (_scroll->scrollTop() == newScrollTop) {
visibleAreaUpdated(); visibleAreaUpdated();
} else { } else {
synteticScrollToY(newScrollTop); synteticScrollToY(toY);
} }
} }
@ -5137,15 +5140,13 @@ bool HistoryWidget::hasPendingResizedItems() const {
} }
std::optional<int> HistoryWidget::unreadBarTop() const { std::optional<int> HistoryWidget::unreadBarTop() const {
auto getUnreadBar = [this]() -> HistoryView::Element* { const auto bar = [&]() -> HistoryView::Element* {
if (const auto bar = _migrated ? _migrated->unreadBar() : nullptr) { if (const auto bar = _migrated ? _migrated->unreadBar() : nullptr) {
return bar; return bar;
} else if (const auto bar = _history->unreadBar()) {
return bar;
} }
return nullptr; return _history->unreadBar();
}; }();
if (const auto bar = getUnreadBar()) { if (bar) {
const auto result = _list->itemTop(bar) const auto result = _list->itemTop(bar)
+ HistoryView::UnreadBar::marginTop(); + HistoryView::UnreadBar::marginTop();
if (bar->Has<HistoryView::DateBadge>()) { if (bar->Has<HistoryView::DateBadge>()) {
@ -5156,15 +5157,6 @@ std::optional<int> HistoryWidget::unreadBarTop() const {
return std::nullopt; return std::nullopt;
} }
HistoryView::Element *HistoryWidget::firstUnreadMessage() const {
if (_migrated) {
if (const auto result = _migrated->firstUnreadMessage()) {
return result;
}
}
return _history ? _history->firstUnreadMessage() : nullptr;
}
void HistoryWidget::addMessagesToFront(PeerData *peer, const QVector<MTPMessage> &messages) { void HistoryWidget::addMessagesToFront(PeerData *peer, const QVector<MTPMessage> &messages) {
_list->messagesReceived(peer, messages); _list->messagesReceived(peer, messages);
if (!_firstLoadRequest) { if (!_firstLoadRequest) {
@ -5180,21 +5172,6 @@ void HistoryWidget::addMessagesToBack(PeerData *peer, const QVector<MTPMessage>
} }
} }
void HistoryWidget::countHistoryShowFrom() {
if (_migrated
&& _showAtMsgId == ShowAtUnreadMsgId
&& _migrated->unreadCount()) {
_migrated->calculateFirstUnreadMessage();
}
if ((_migrated && _migrated->firstUnreadMessage())
|| (_showAtMsgId != ShowAtUnreadMsgId)
|| !_history->unreadCount()) {
_history->unsetFirstUnreadMessage();
} else {
_history->calculateFirstUnreadMessage();
}
}
void HistoryWidget::updateBotKeyboard(History *h, bool force) { void HistoryWidget::updateBotKeyboard(History *h, bool force) {
if (h && h != _history && h != _migrated) { if (h && h != _history && h != _migrated) {
return; return;
@ -5311,7 +5288,7 @@ void HistoryWidget::updateHistoryDownPosition() {
void HistoryWidget::updateHistoryDownVisibility() { void HistoryWidget::updateHistoryDownVisibility() {
if (_a_show.animating()) return; if (_a_show.animating()) return;
auto haveUnreadBelowBottom = [&](History *history) { const auto haveUnreadBelowBottom = [&](History *history) {
if (!_list || !history || history->unreadCount() <= 0) { if (!_list || !history || history->unreadCount() <= 0) {
return false; return false;
} }
@ -5322,7 +5299,7 @@ void HistoryWidget::updateHistoryDownVisibility() {
const auto top = _list->itemTop(unread); const auto top = _list->itemTop(unread);
return (top >= _scroll->scrollTop() + _scroll->height()); return (top >= _scroll->scrollTop() + _scroll->height());
}; };
auto historyDownIsVisible = [&] { const auto historyDownIsVisible = [&] {
if (!_list || _firstLoadRequest) { if (!_list || _firstLoadRequest) {
return false; return false;
} }

View File

@ -557,6 +557,7 @@ private:
// destroys _history and _migrated unread bars // destroys _history and _migrated unread bars
void destroyUnreadBar(); void destroyUnreadBar();
void destroyUnreadBarOnClose();
void saveEditMsg(); void saveEditMsg();
void saveEditMsgDone(History *history, const MTPUpdates &updates, mtpRequestId req); void saveEditMsgDone(History *history, const MTPUpdates &updates, mtpRequestId req);
@ -583,15 +584,12 @@ private:
std::optional<int> unreadBarTop() const; std::optional<int> unreadBarTop() const;
int itemTopForHighlight(not_null<HistoryView::Element*> view) const; int itemTopForHighlight(not_null<HistoryView::Element*> view) const;
void scrollToCurrentVoiceMessage(FullMsgId fromId, FullMsgId toId); void scrollToCurrentVoiceMessage(FullMsgId fromId, FullMsgId toId);
HistoryView::Element *firstUnreadMessage() const;
// Scroll to current y without updating the _lastUserScrolled time. // Scroll to current y without updating the _lastUserScrolled time.
// Used to distinguish between user scrolls and syntetic scrolls. // Used to distinguish between user scrolls and syntetic scrolls.
// This one is syntetic. // This one is syntetic.
void synteticScrollToY(int y); void synteticScrollToY(int y);
void countHistoryShowFrom();
void writeDrafts(Data::Draft **localDraft, Data::Draft **editDraft); void writeDrafts(Data::Draft **localDraft, Data::Draft **editDraft);
void writeDrafts(History *history); void writeDrafts(History *history);
void setFieldText( void setFieldText(