Display active feed state in dialogs list.

This commit is contained in:
John Preston 2018-01-22 19:42:25 +03:00
parent 840b42934b
commit 47ad5ea98a
29 changed files with 381 additions and 250 deletions

View File

@ -108,7 +108,10 @@ DialogsInner::DialogsInner(QWidget *parent, not_null<Window::Controller*> contro
}, lifetime()); }, lifetime());
subscribe(App::histories().sendActionAnimationUpdated(), [this](const Histories::SendActionAnimationUpdate &update) { subscribe(App::histories().sendActionAnimationUpdated(), [this](const Histories::SendActionAnimationUpdate &update) {
auto updateRect = Dialogs::Layout::RowPainter::sendActionAnimationRect(update.width, update.height, getFullWidth(), update.textUpdated); auto updateRect = Dialogs::Layout::RowPainter::sendActionAnimationRect(update.width, update.height, getFullWidth(), update.textUpdated);
updateDialogRow(update.history, MsgId(0), updateRect, UpdateRowSection::Default | UpdateRowSection::Filtered); updateDialogRow(
Dialogs::RowDescriptor(update.history, MsgId(0)),
updateRect,
UpdateRowSection::Default | UpdateRowSection::Filtered);
}); });
subscribe(Window::Theme::Background(), [this](const Window::Theme::BackgroundUpdate &data) { subscribe(Window::Theme::Background(), [this](const Window::Theme::BackgroundUpdate &data) {
@ -140,6 +143,15 @@ DialogsInner::DialogsInner(QWidget *parent, not_null<Window::Controller*> contro
} }
})); }));
_controller->activeChatEntryValue(
) | rpl::combine_previous(
) | rpl::start_with_next([=](
Dialogs::RowDescriptor previous,
Dialogs::RowDescriptor next) {
const auto rect = QRect(0, 0, getFullWidth(), st::dialogsRowHeight);
updateDialogRow(previous, rect);
updateDialogRow(next, rect);
}, lifetime());
refresh(); refresh();
} }
@ -181,6 +193,7 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
if (!paintingOther) { if (!paintingOther) {
p.setClipRect(r); p.setClipRect(r);
} }
const auto activeEntry = _controller->activeChatEntryCurrent();
auto fullWidth = getFullWidth(); auto fullWidth = getFullWidth();
auto ms = getms(); auto ms = getms();
if (_state == State::Default) { if (_state == State::Default) {
@ -195,15 +208,7 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
p.translate(0, st::dialogsImportantBarHeight); p.translate(0, st::dialogsImportantBarHeight);
} }
auto otherStart = rows->size() * st::dialogsRowHeight; auto otherStart = rows->size() * st::dialogsRowHeight;
// #TODO feeds show const auto active = activeEntry.key;
const auto active = [&] {
if (const auto peer = App::main()->activePeer()) {
if (const auto history = App::historyLoaded(peer)) {
return Dialogs::Key(history);
}
}
return Dialogs::Key();
}();
const auto selected = _menuKey const auto selected = _menuKey
? _menuKey ? _menuKey
: (isPressed() : (isPressed()
@ -238,7 +243,14 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
// Skip currently dragged chat to paint it above others after. // Skip currently dragged chat to paint it above others after.
if (lastPaintedPos != _aboveIndex) { if (lastPaintedPos != _aboveIndex) {
paintDialog(p, row, fullWidth, active, selected, paintingOther, ms); paintDialog(
p,
row,
fullWidth,
active,
selected,
paintingOther,
ms);
} }
p.translate(0, st::dialogsRowHeight); p.translate(0, st::dialogsRowHeight);
@ -318,19 +330,11 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
auto to = ceilclamp(r.y() + r.height() - skip, st::dialogsRowHeight, 0, _filterResults.size()); auto to = ceilclamp(r.y() + r.height() - skip, st::dialogsRowHeight, 0, _filterResults.size());
p.translate(0, from * st::dialogsRowHeight); p.translate(0, from * st::dialogsRowHeight);
if (from < _filterResults.size()) { if (from < _filterResults.size()) {
// #TODO feeds show
const auto activePeer = App::main()->activePeer();
const auto activeMsgId = App::main()->activeMsgId();
for (; from < to; ++from) { for (; from < to; ++from) {
const auto row = _filterResults[from]; const auto row = _filterResults[from];
const auto key = row->key(); const auto key = row->key();
const auto history = key.history(); const auto active = (activeEntry.key == key)
const auto peer = history ? history->peer.get() : nullptr; && !activeEntry.msgId;
const auto active = !activeMsgId
&& peer
&& activePeer
&& ((peer == activePeer)
|| (peer->migrateTo() == activePeer));
const auto selected = _menuKey const auto selected = _menuKey
? (key == _menuKey) ? (key == _menuKey)
: (from == (isPressed() : (from == (isPressed()
@ -363,12 +367,11 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
auto to = ceilclamp(r.y() + r.height() - skip, st::dialogsRowHeight, 0, _peerSearchResults.size()); auto to = ceilclamp(r.y() + r.height() - skip, st::dialogsRowHeight, 0, _peerSearchResults.size());
p.translate(0, from * st::dialogsRowHeight); p.translate(0, from * st::dialogsRowHeight);
if (from < _peerSearchResults.size()) { if (from < _peerSearchResults.size()) {
auto activePeer = App::main()->activePeer(); const auto activePeer = activeEntry.key.peer();
auto activeMsgId = App::main()->activeMsgId();
for (; from < to; ++from) { for (; from < to; ++from) {
const auto &result = _peerSearchResults[from]; const auto &result = _peerSearchResults[from];
const auto peer = result->peer; const auto peer = result->peer;
const auto active = !activeMsgId const auto active = !activeEntry.msgId
&& activePeer && activePeer
&& ((peer == activePeer) && ((peer == activePeer)
|| (peer->migrateTo() == activePeer)); || (peer->migrateTo() == activePeer));
@ -414,21 +417,27 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
auto to = ceilclamp(r.y() + r.height() - skip, st::dialogsRowHeight, 0, _searchResults.size()); auto to = ceilclamp(r.y() + r.height() - skip, st::dialogsRowHeight, 0, _searchResults.size());
p.translate(0, from * st::dialogsRowHeight); p.translate(0, from * st::dialogsRowHeight);
if (from < _searchResults.size()) { if (from < _searchResults.size()) {
auto activePeer = App::main()->activePeer(); const auto activePeer = activeEntry.key.peer();
auto activeMsgId = App::main()->activeMsgId();
for (; from < to; ++from) { for (; from < to; ++from) {
const auto &result = _searchResults[from]; const auto &result = _searchResults[from];
const auto item = result->item(); const auto item = result->item();
const auto peer = item->history()->peer; const auto peer = item->history()->peer;
const auto active = (peer == activePeer const auto active = (peer == activePeer
&& item->id == activeMsgId) && item->id == activeEntry.msgId)
|| (peer->migrateTo() || (peer->migrateTo()
&& peer->migrateTo() == activePeer && peer->migrateTo() == activePeer
&& item->id == -activeMsgId); && item->id == -activeEntry.msgId);
const auto selected = (from == (isPressed() const auto selected = (from == (isPressed()
? _searchedPressed ? _searchedPressed
: _searchedSelected)); : _searchedSelected));
Dialogs::Layout::RowPainter::paint(p, result.get(), fullWidth, active, selected, paintingOther, ms); Dialogs::Layout::RowPainter::paint(
p,
result.get(),
fullWidth,
active,
selected,
paintingOther,
ms);
p.translate(0, st::dialogsRowHeight); p.translate(0, st::dialogsRowHeight);
} }
} }
@ -1243,8 +1252,7 @@ void DialogsInner::dlgUpdated(
void DialogsInner::dlgUpdated(not_null<History*> history, MsgId msgId) { void DialogsInner::dlgUpdated(not_null<History*> history, MsgId msgId) {
updateDialogRow( updateDialogRow(
history, Dialogs::RowDescriptor(history, msgId),
msgId,
QRect(0, 0, getFullWidth(), st::dialogsRowHeight)); QRect(0, 0, getFullWidth(), st::dialogsRowHeight));
} }
@ -1264,17 +1272,20 @@ void DialogsInner::updateSearchResult(not_null<PeerData*> peer) {
} }
void DialogsInner::updateDialogRow( void DialogsInner::updateDialogRow(
not_null<History*> history, Dialogs::RowDescriptor row,
MsgId msgId,
QRect updateRect, QRect updateRect,
UpdateRowSections sections) { UpdateRowSections sections) {
auto updateRow = [this, updateRect](int rowTop) { auto updateRow = [&](int rowTop) {
rtlupdate(updateRect.x(), rowTop + updateRect.y(), updateRect.width(), updateRect.height()); rtlupdate(
updateRect.x(),
rowTop + updateRect.y(),
updateRect.width(),
updateRect.height());
}; };
if (_state == State::Default) { if (_state == State::Default) {
if (sections & UpdateRowSection::Default) { if (sections & UpdateRowSection::Default) {
if (const auto row = shownDialogs()->getRow(history)) { if (const auto dialog = shownDialogs()->getRow(row.key)) {
auto position = row->pos(); const auto position = dialog->pos();
auto top = dialogsOffset(); auto top = dialogsOffset();
if (base::in_range(position, 0, _pinnedRows.size())) { if (base::in_range(position, 0, _pinnedRows.size())) {
top += qRound(_pinnedRows[position].yadd.current()); top += qRound(_pinnedRows[position].yadd.current());
@ -1283,35 +1294,46 @@ void DialogsInner::updateDialogRow(
} }
} }
} else if (_state == State::Filtered) { } else if (_state == State::Filtered) {
if ((sections & UpdateRowSection::Filtered) && !_filterResults.isEmpty()) { if ((sections & UpdateRowSection::Filtered)
auto index = 0, add = filteredOffset(); && !_filterResults.isEmpty()) {
for_const (auto row, _filterResults) { const auto add = filteredOffset();
if (row->history() == history) { auto index = 0;
for (const auto result : _filterResults) {
if (result->key() == row.key) {
updateRow(add + index * st::dialogsRowHeight); updateRow(add + index * st::dialogsRowHeight);
break; break;
} }
++index; ++index;
} }
} }
if ((sections & UpdateRowSection::PeerSearch) && !_peerSearchResults.empty()) { if ((sections & UpdateRowSection::PeerSearch)
auto index = 0, add = peerSearchOffset(); && !_peerSearchResults.empty()) {
for_const (auto &result, _peerSearchResults) { if (const auto peer = row.key.peer()) {
if (result->peer == history->peer) { const auto add = peerSearchOffset();
updateRow(add + index * st::dialogsRowHeight); auto index = 0;
break; for (const auto &result : _peerSearchResults) {
if (result->peer == peer) {
updateRow(add + index * st::dialogsRowHeight);
break;
}
++index;
} }
++index;
} }
} }
if ((sections & UpdateRowSection::MessageSearch) && !_searchResults.empty()) { if ((sections & UpdateRowSection::MessageSearch)
auto index = 0, add = searchedOffset(); && !_searchResults.empty()) {
for_const (auto &result, _searchResults) { if (const auto history = row.key.history()) {
auto item = result->item(); const auto add = searchedOffset();
if (item->history() == history && item->id == msgId) { auto index = 0;
updateRow(add + index * st::dialogsRowHeight); for (const auto &result : _searchResults) {
break; auto item = result->item();
if (item->history() == history
&& item->id == row.msgId) {
updateRow(add + index * st::dialogsRowHeight);
break;
}
++index;
} }
++index;
} }
} }
} }

View File

@ -207,8 +207,7 @@ private:
void updateSearchResult(not_null<PeerData*> peer); void updateSearchResult(not_null<PeerData*> peer);
void updateDialogRow( void updateDialogRow(
not_null<History*> history, Dialogs::RowDescriptor row,
MsgId msgId,
QRect updateRect, QRect updateRect,
UpdateRowSections sections = UpdateRowSection::All); UpdateRowSections sections = UpdateRowSection::All);

View File

@ -35,4 +35,11 @@ Data::Feed *Key::feed() const {
return nullptr; return nullptr;
} }
PeerData *Key::peer() const {
if (const auto p = base::get_if<not_null<History*>>(&_value)) {
return (*p)->peer;
}
return nullptr;
}
} // namespace Dialogs } // namespace Dialogs

View File

@ -37,6 +37,7 @@ public:
not_null<Entry*> entry() const; not_null<Entry*> entry() const;
History *history() const; History *history() const;
Data::Feed *feed() const; Data::Feed *feed() const;
PeerData *peer() const;
inline bool operator<(const Key &other) const { inline bool operator<(const Key &other) const {
return _value < other._value; return _value < other._value;
@ -80,6 +81,36 @@ struct RowDescriptor {
Key key; Key key;
MsgId msgId = 0; MsgId msgId = 0;
}; };
inline bool operator==(const RowDescriptor &a, const RowDescriptor &b) {
return (a.key == b.key) && (a.msgId == b.msgId);
}
inline bool operator!=(const RowDescriptor &a, const RowDescriptor &b) {
return !(a == b);
}
inline bool operator<(const RowDescriptor &a, const RowDescriptor &b) {
if (a.key < b.key) {
return true;
} else if (a.key > b.key) {
return false;
}
return a.msgId < b.msgId;
}
inline bool operator>(const RowDescriptor &a, const RowDescriptor &b) {
return (b < a);
}
inline bool operator<=(const RowDescriptor &a, const RowDescriptor &b) {
return !(b < a);
}
inline bool operator>=(const RowDescriptor &a, const RowDescriptor &b) {
return !(a < b);
}
} // namespace Dialogs } // namespace Dialogs

View File

@ -210,7 +210,11 @@ InnerWidget::InnerWidget(
, _channel(channel) , _channel(channel)
, _history(App::history(channel)) , _history(App::history(channel))
, _scrollDateCheck([this] { scrollDateCheck(); }) , _scrollDateCheck([this] { scrollDateCheck(); })
, _emptyText(st::historyAdminLogEmptyWidth - st::historyAdminLogEmptyPadding.left() - st::historyAdminLogEmptyPadding.left()) { , _emptyText(
st::historyAdminLogEmptyWidth
- st::historyAdminLogEmptyPadding.left()
- st::historyAdminLogEmptyPadding.left())
, _idManager(_history->adminLogIdManager()) {
setMouseTracking(true); setMouseTracking(true);
_scrollDateHideTimer.setCallback([this] { scrollDateHideByTimer(); }); _scrollDateHideTimer.setCallback([this] { scrollDateHideByTimer(); });
Auth().data().viewRepaintRequest( Auth().data().viewRepaintRequest(
@ -511,7 +515,9 @@ void InnerWidget::saveState(not_null<SectionMemento*> memento) {
void InnerWidget::restoreState(not_null<SectionMemento*> memento) { void InnerWidget::restoreState(not_null<SectionMemento*> memento) {
_items = memento->takeItems(); _items = memento->takeItems();
_itemsByIds = memento->takeItemsByIds(); _itemsByIds = memento->takeItemsByIds();
_idManager = memento->takeIdManager(); if (auto manager = memento->takeIdManager()) {
_idManager = std::move(manager);
}
_admins = memento->takeAdmins(); _admins = memento->takeAdmins();
_adminsCanEdit = memento->takeAdminsCanEdit(); _adminsCanEdit = memento->takeAdminsCanEdit();
_filter = memento->takeFilter(); _filter = memento->takeFilter();
@ -601,7 +607,7 @@ void InnerWidget::addEvents(Direction direction, const QVector<MTPChannelAdminLo
GenerateItems( GenerateItems(
this, this,
_history, _history,
_idManager, _idManager.get(),
data, data,
addOne); addOne);
if (count > 1) { if (count > 1) {
@ -805,7 +811,8 @@ void InnerWidget::clearAfterFilterChange() {
_filterChanged = false; _filterChanged = false;
_items.clear(); _items.clear();
_itemsByIds.clear(); _itemsByIds.clear();
_idManager = LocalIdManager(); _idManager = nullptr;
_idManager = _history->adminLogIdManager();
updateEmptyText(); updateEmptyText();
updateSize(); updateSize();
} }

View File

@ -198,7 +198,6 @@ private:
int _itemsWidth = 0; int _itemsWidth = 0;
int _itemsHeight = 0; int _itemsHeight = 0;
LocalIdManager _idManager;
int _minHeight = 0; int _minHeight = 0;
int _visibleTop = 0; int _visibleTop = 0;
int _visibleBottom = 0; int _visibleBottom = 0;
@ -251,6 +250,8 @@ private:
std::vector<not_null<UserData*>> _adminsCanEdit; std::vector<not_null<UserData*>> _adminsCanEdit;
base::lambda<void(FilterValue &&filter)> _showFilterCallback; base::lambda<void(FilterValue &&filter)> _showFilterCallback;
std::shared_ptr<LocalIdManager> _idManager;
}; };
} // namespace AdminLog } // namespace AdminLog

View File

@ -312,7 +312,7 @@ OwnedItem::~OwnedItem() {
void GenerateItems( void GenerateItems(
not_null<HistoryView::ElementDelegate*> delegate, not_null<HistoryView::ElementDelegate*> delegate,
not_null<History*> history, not_null<History*> history,
LocalIdManager &idManager, not_null<LocalIdManager*> idManager,
const MTPDchannelAdminLogEvent &event, const MTPDchannelAdminLogEvent &event,
base::lambda<void(OwnedItem item)> callback) { base::lambda<void(OwnedItem item)> callback) {
Expects(history->peer->isChannel()); Expects(history->peer->isChannel());
@ -334,7 +334,7 @@ void GenerateItems(
auto addSimpleServiceMessage = [&](const QString &text, PhotoData *photo = nullptr) { auto addSimpleServiceMessage = [&](const QString &text, PhotoData *photo = nullptr) {
auto message = HistoryService::PreparedText { text }; auto message = HistoryService::PreparedText { text };
message.links.push_back(fromLink); message.links.push_back(fromLink);
addPart(new HistoryService(history, idManager.next(), ::date(date), message, 0, peerToUser(from->id), photo)); addPart(new HistoryService(history, idManager->next(), ::date(date), message, 0, peerToUser(from->id), photo));
}; };
auto createChangeTitle = [&](const MTPDchannelAdminLogEventActionChangeTitle &action) { auto createChangeTitle = [&](const MTPDchannelAdminLogEventActionChangeTitle &action) {
@ -355,7 +355,7 @@ void GenerateItems(
auto bodyReplyTo = 0; auto bodyReplyTo = 0;
auto bodyViaBotId = 0; auto bodyViaBotId = 0;
auto newDescription = PrepareText(newValue, QString()); auto newDescription = PrepareText(newValue, QString());
auto body = new HistoryMessage(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), newDescription); auto body = new HistoryMessage(history, idManager->next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), newDescription);
if (!oldValue.isEmpty()) { if (!oldValue.isEmpty()) {
auto oldDescription = PrepareText(oldValue, QString()); auto oldDescription = PrepareText(oldValue, QString());
body->addLogEntryOriginal(id, lang(lng_admin_log_previous_description), oldDescription); body->addLogEntryOriginal(id, lang(lng_admin_log_previous_description), oldDescription);
@ -376,7 +376,7 @@ void GenerateItems(
auto bodyReplyTo = 0; auto bodyReplyTo = 0;
auto bodyViaBotId = 0; auto bodyViaBotId = 0;
auto newLink = newValue.isEmpty() ? TextWithEntities() : PrepareText(Messenger::Instance().createInternalLinkFull(newValue), QString()); auto newLink = newValue.isEmpty() ? TextWithEntities() : PrepareText(Messenger::Instance().createInternalLinkFull(newValue), QString());
auto body = new HistoryMessage(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), newLink); auto body = new HistoryMessage(history, idManager->next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), newLink);
if (!oldValue.isEmpty()) { if (!oldValue.isEmpty()) {
auto oldLink = PrepareText(Messenger::Instance().createInternalLinkFull(oldValue), QString()); auto oldLink = PrepareText(Messenger::Instance().createInternalLinkFull(oldValue), QString());
body->addLogEntryOriginal(id, lang(lng_admin_log_previous_link), oldLink); body->addLogEntryOriginal(id, lang(lng_admin_log_previous_link), oldLink);
@ -427,7 +427,7 @@ void GenerateItems(
addPart(history->createItem( addPart(history->createItem(
PrepareLogMessage( PrepareLogMessage(
action.vmessage, action.vmessage,
idManager.next(), idManager->next(),
date.v), date.v),
detachExistingItem)); detachExistingItem));
} }
@ -447,7 +447,7 @@ void GenerateItems(
auto body = history->createItem( auto body = history->createItem(
PrepareLogMessage( PrepareLogMessage(
action.vnew_message, action.vnew_message,
idManager.next(), idManager->next(),
date.v), date.v),
detachExistingItem); detachExistingItem);
if (oldValue.text.isEmpty()) { if (oldValue.text.isEmpty()) {
@ -463,7 +463,7 @@ void GenerateItems(
auto detachExistingItem = false; auto detachExistingItem = false;
addPart(history->createItem( addPart(history->createItem(
PrepareLogMessage(action.vmessage, idManager.next(), date.v), PrepareLogMessage(action.vmessage, idManager->next(), date.v),
detachExistingItem)); detachExistingItem));
}; };
@ -486,7 +486,7 @@ void GenerateItems(
auto bodyReplyTo = 0; auto bodyReplyTo = 0;
auto bodyViaBotId = 0; auto bodyViaBotId = 0;
auto bodyText = GenerateParticipantChangeText(channel, action.vparticipant); auto bodyText = GenerateParticipantChangeText(channel, action.vparticipant);
addPart(new HistoryMessage(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), bodyText)); addPart(new HistoryMessage(history, idManager->next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), bodyText));
}; };
auto createParticipantToggleBan = [&](const MTPDchannelAdminLogEventActionParticipantToggleBan &action) { auto createParticipantToggleBan = [&](const MTPDchannelAdminLogEventActionParticipantToggleBan &action) {
@ -494,7 +494,7 @@ void GenerateItems(
auto bodyReplyTo = 0; auto bodyReplyTo = 0;
auto bodyViaBotId = 0; auto bodyViaBotId = 0;
auto bodyText = GenerateParticipantChangeText(channel, action.vnew_participant, &action.vprev_participant); auto bodyText = GenerateParticipantChangeText(channel, action.vnew_participant, &action.vprev_participant);
addPart(new HistoryMessage(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), bodyText)); addPart(new HistoryMessage(history, idManager->next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), bodyText));
}; };
auto createParticipantToggleAdmin = [&](const MTPDchannelAdminLogEventActionParticipantToggleAdmin &action) { auto createParticipantToggleAdmin = [&](const MTPDchannelAdminLogEventActionParticipantToggleAdmin &action) {
@ -502,7 +502,7 @@ void GenerateItems(
auto bodyReplyTo = 0; auto bodyReplyTo = 0;
auto bodyViaBotId = 0; auto bodyViaBotId = 0;
auto bodyText = GenerateParticipantChangeText(channel, action.vnew_participant, &action.vprev_participant); auto bodyText = GenerateParticipantChangeText(channel, action.vnew_participant, &action.vprev_participant);
addPart(new HistoryMessage(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), bodyText)); addPart(new HistoryMessage(history, idManager->next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date), peerToUser(from->id), QString(), bodyText));
}; };
auto createChangeStickerSet = [&](const MTPDchannelAdminLogEventActionChangeStickerSet &action) { auto createChangeStickerSet = [&](const MTPDchannelAdminLogEventActionChangeStickerSet &action) {
@ -523,7 +523,7 @@ void GenerateItems(
auto message = HistoryService::PreparedText { text }; auto message = HistoryService::PreparedText { text };
message.links.push_back(fromLink); message.links.push_back(fromLink);
message.links.push_back(setLink); message.links.push_back(setLink);
addPart(new HistoryService(history, idManager.next(), ::date(date), message, 0, peerToUser(from->id))); addPart(new HistoryService(history, idManager->next(), ::date(date), message, 0, peerToUser(from->id)));
} }
}; };

View File

@ -20,7 +20,7 @@ class LocalIdManager;
void GenerateItems( void GenerateItems(
not_null<HistoryView::ElementDelegate*> delegate, not_null<HistoryView::ElementDelegate*> delegate,
not_null<History*> history, not_null<History*> history,
LocalIdManager &idManager, not_null<LocalIdManager*> idManager,
const MTPDchannelAdminLogEvent &event, const MTPDchannelAdminLogEvent &event,
base::lambda<void(OwnedItem item)> callback); base::lambda<void(OwnedItem item)> callback);

View File

@ -279,6 +279,10 @@ not_null<ChannelData*> Widget::channel() const {
return _inner->channel(); return _inner->channel();
} }
Dialogs::RowDescriptor Widget::activeChat() const {
return { App::history(channel()), MsgId(0) };
}
QPixmap Widget::grabForShowAnimation(const Window::SectionSlideParams &params) { QPixmap Widget::grabForShowAnimation(const Window::SectionSlideParams &params) {
if (params.withTopBarShadow) _fixedBarShadow->hide(); if (params.withTopBarShadow) _fixedBarShadow->hide();
auto result = Ui::GrabWidget(this); auto result = Ui::GrabWidget(this);

View File

@ -52,12 +52,6 @@ public:
LocalIdManager() = default; LocalIdManager() = default;
LocalIdManager(const LocalIdManager &other) = delete; LocalIdManager(const LocalIdManager &other) = delete;
LocalIdManager &operator=(const LocalIdManager &other) = delete; LocalIdManager &operator=(const LocalIdManager &other) = delete;
LocalIdManager(LocalIdManager &&other) : _counter(std::exchange(other._counter, ServerMaxMsgId)) {
}
LocalIdManager &operator=(LocalIdManager &&other) {
_counter = std::exchange(other._counter, ServerMaxMsgId);
return *this;
}
MsgId next() { MsgId next() {
return ++_counter; return ++_counter;
} }
@ -72,9 +66,7 @@ public:
Widget(QWidget *parent, not_null<Window::Controller*> controller, not_null<ChannelData*> channel); Widget(QWidget *parent, not_null<Window::Controller*> controller, not_null<ChannelData*> channel);
not_null<ChannelData*> channel() const; not_null<ChannelData*> channel() const;
PeerData *activePeer() const override { Dialogs::RowDescriptor activeChat() const override;
return channel();
}
bool hasTopBarShadow() const override { bool hasTopBarShadow() const override {
return true; return true;
@ -173,7 +165,7 @@ public:
void setSearchQuery(QString &&query) { void setSearchQuery(QString &&query) {
_searchQuery = std::move(query); _searchQuery = std::move(query);
} }
void setIdManager(LocalIdManager &&manager) { void setIdManager(std::shared_ptr<LocalIdManager> &&manager) {
_idManager = std::move(manager); _idManager = std::move(manager);
} }
std::vector<OwnedItem> takeItems() { std::vector<OwnedItem> takeItems() {
@ -182,7 +174,7 @@ public:
std::map<uint64, not_null<Element*>> takeItemsByIds() { std::map<uint64, not_null<Element*>> takeItemsByIds() {
return std::move(_itemsByIds); return std::move(_itemsByIds);
} }
LocalIdManager takeIdManager() { std::shared_ptr<LocalIdManager> takeIdManager() {
return std::move(_idManager); return std::move(_idManager);
} }
bool upLoaded() const { bool upLoaded() const {
@ -207,7 +199,7 @@ private:
std::map<uint64, not_null<Element*>> _itemsByIds; std::map<uint64, not_null<Element*>> _itemsByIds;
bool _upLoaded = false; bool _upLoaded = false;
bool _downLoaded = true; bool _downLoaded = true;
LocalIdManager _idManager; std::shared_ptr<LocalIdManager> _idManager;
FilterValue _filter; FilterValue _filter;
QString _searchQuery; QString _searchQuery;

View File

@ -93,6 +93,10 @@ Widget::Widget(
}, lifetime()); }, lifetime());
} }
Dialogs::RowDescriptor Widget::activeChat() const {
return Dialogs::RowDescriptor(_feed, MsgId(0));
}
void Widget::updateAdaptiveLayout() { void Widget::updateAdaptiveLayout() {
_topBarShadow->moveToLeft( _topBarShadow->moveToLeft(
Adaptive::OneColumn() ? 0 : st::lineWidth, Adaptive::OneColumn() ? 0 : st::lineWidth,

View File

@ -36,6 +36,8 @@ public:
not_null<Window::Controller*> controller, not_null<Window::Controller*> controller,
not_null<Data::Feed*> feed); not_null<Data::Feed*> feed);
Dialogs::RowDescriptor activeChat() const override;
bool hasTopBarShadow() const override { bool hasTopBarShadow() const override {
return true; return true;
} }

View File

@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history.h" #include "history/history.h"
#include "history/view/history_view_element.h" #include "history/view/history_view_element.h"
#include "history/admin_log/history_admin_log_section.h"
#include "history/history_message.h" #include "history/history_message.h"
#include "history/history_media_types.h" #include "history/history_media_types.h"
#include "history/history_service.h" #include "history/history_service.h"
@ -1924,6 +1925,15 @@ void History::getNextFirstUnreadMessage() {
_firstUnreadView = nullptr; _firstUnreadView = nullptr;
} }
std::shared_ptr<AdminLog::LocalIdManager> History::adminLogIdManager() {
if (const auto strong = _adminLogIdManager.lock()) {
return strong;
}
auto result = std::make_shared<AdminLog::LocalIdManager>();
_adminLogIdManager = result;
return result;
}
QDateTime History::adjustChatListDate() const { QDateTime History::adjustChatListDate() const {
const auto result = chatsListDate(); const auto result = chatsListDate();
if (const auto draft = cloudDraft()) { if (const auto draft = cloudDraft()) {

View File

@ -19,16 +19,20 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
class History; class History;
namespace HistoryView {
class Element;
} // namespace HistoryView
namespace AdminLog {
class LocalIdManager;
} // namespace AdminLog
enum NewMessageType : char { enum NewMessageType : char {
NewMessageUnread, NewMessageUnread,
NewMessageLast, NewMessageLast,
NewMessageExisting, NewMessageExisting,
}; };
namespace HistoryView {
class Element;
} // namespace HistoryView
class Histories { class Histories {
public: public:
using Map = QHash<PeerId, History*>; using Map = QHash<PeerId, History*>;
@ -331,6 +335,8 @@ public:
// of the displayed window relative to the history start coordinate // of the displayed window relative to the history start coordinate
void countScrollState(int top); void countScrollState(int top);
std::shared_ptr<AdminLog::LocalIdManager> adminLogIdManager();
virtual ~History(); virtual ~History();
// Still public data. // Still public data.
@ -503,6 +509,8 @@ private:
int _pinnedIndex = 0; // > 0 for pinned dialogs int _pinnedIndex = 0; // > 0 for pinned dialogs
std::weak_ptr<AdminLog::LocalIdManager> _adminLogIdManager;
}; };
class HistoryJoined; class HistoryJoined;

View File

@ -1371,7 +1371,9 @@ bool HistoryWidget::notify_switchInlineBotButtonReceived(const QString &query, U
return true; return true;
} }
} else if (auto bot = _peer ? _peer->asUser() : nullptr) { } else if (auto bot = _peer ? _peer->asUser() : nullptr) {
PeerId toPeerId = bot->botInfo ? bot->botInfo->inlineReturnPeerId : 0; const auto toPeerId = bot->botInfo
? bot->botInfo->inlineReturnPeerId
: PeerId(0);
if (!toPeerId) { if (!toPeerId) {
return false; return false;
} }
@ -1620,11 +1622,6 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
bool canShowNow = _history->isReadyFor(showAtMsgId); bool canShowNow = _history->isReadyFor(showAtMsgId);
if (!canShowNow) { if (!canShowNow) {
delayedShowAt(showAtMsgId); delayedShowAt(showAtMsgId);
if (wasHistory) {
App::main()->dlgUpdated(wasHistory, wasMsgId);
}
emit historyShown(_history, _showAtMsgId);
} else { } else {
_history->forgetScrollState(); _history->forgetScrollState();
if (_migrated) { if (_migrated) {
@ -1655,12 +1652,18 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
_topBar->update(); _topBar->update();
update(); update();
if (startBot && _peer->isUser() && _peer->asUser()->botInfo) { if (const auto user = _peer->asUser()) {
if (wasHistory) _peer->asUser()->botInfo->inlineReturnPeerId = wasHistory->peer->id; if (const auto &info = user->botInfo) {
onBotStart(); if (startBot) {
_history->clearLocalDraft(); if (wasHistory) {
applyDraft(); info->inlineReturnPeerId = wasHistory->peer->id;
_send->finishAnimating(); }
onBotStart();
_history->clearLocalDraft();
applyDraft();
_send->finishAnimating();
}
}
} }
return; return;
} }
@ -1802,9 +1805,15 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
connect(_scroll, SIGNAL(geometryChanged()), _list, SLOT(onParentGeometryChanged())); connect(_scroll, SIGNAL(geometryChanged()), _list, SLOT(onParentGeometryChanged()));
if (startBot && _peer->isUser() && _peer->asUser()->botInfo) { if (const auto user = _peer->asUser()) {
if (wasHistory) _peer->asUser()->botInfo->inlineReturnPeerId = wasHistory->peer->id; if (const auto &info = user->botInfo) {
onBotStart(); if (startBot) {
if (wasHistory) {
info->inlineReturnPeerId = wasHistory->peer->id;
}
onBotStart();
}
}
} }
unreadCountChanged(_history); // set _historyDown badge. unreadCountChanged(_history); // set _historyDown badge.
} else { } else {
@ -1815,16 +1824,11 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
updateForwarding(); updateForwarding();
updateOverStates(mapFromGlobal(QCursor::pos())); updateOverStates(mapFromGlobal(QCursor::pos()));
if (App::wnd()) QTimer::singleShot(0, App::wnd(), SLOT(setInnerFocus())); crl::on_main(App::wnd(), [] { App::wnd()->setInnerFocus(); });
if (wasHistory) {
App::main()->dlgUpdated(wasHistory, wasMsgId);
}
emit historyShown(_history, _showAtMsgId);
controller()->historyPeer = _peer; controller()->historyPeer = _peer;
if (_peer) { if (_history) {
controller()->activePeer = _peer; controller()->setActiveChatEntry({ _history, _showAtMsgId });
} }
update(); update();
} }
@ -3038,9 +3042,8 @@ void HistoryWidget::setMsgId(MsgId showAtMsgId) {
auto wasMsgId = _showAtMsgId; auto wasMsgId = _showAtMsgId;
_showAtMsgId = showAtMsgId; _showAtMsgId = showAtMsgId;
if (_history) { if (_history) {
App::main()->dlgUpdated(_history, wasMsgId); controller()->setActiveChatEntry({ _history, _showAtMsgId });
} }
emit historyShown(_history, _showAtMsgId);
} }
} }

View File

@ -367,7 +367,6 @@ protected:
signals: signals:
void cancelled(); void cancelled();
void historyShown(History *history, MsgId atMsgId);
public slots: public slots:
void onCancel(); void onCancel();

View File

@ -1304,6 +1304,9 @@ bool Message::hasBubble() const {
} }
bool Message::hasFastReply() const { bool Message::hasFastReply() const {
if (context() != Context::History) {
return false;
}
const auto peer = data()->history()->peer; const auto peer = data()->history()->peer;
return !hasOutLayout() && (peer->isChat() || peer->isMegagroup()); return !hasOutLayout() && (peer->isChat() || peer->isMegagroup());
} }
@ -1491,23 +1494,24 @@ TextSelection Message::unskipTextSelection(TextSelection selection) const {
QRect Message::countGeometry() const { QRect Message::countGeometry() const {
const auto item = message(); const auto item = message();
const auto media = this->media();
const auto mediaWidth = media ? media->width() : width();
const auto outbg = hasOutLayout(); const auto outbg = hasOutLayout();
const auto availableWidth = width()
- st::msgMargin.left()
- st::msgMargin.right();
auto contentLeft = (outbg && !Adaptive::ChatWide()) auto contentLeft = (outbg && !Adaptive::ChatWide())
? st::msgMargin.right() ? st::msgMargin.right()
: st::msgMargin.left(); : st::msgMargin.left();
auto contentWidth = availableWidth;
if (hasFromPhoto()) { if (hasFromPhoto()) {
contentLeft += st::msgPhotoSkip; contentLeft += st::msgPhotoSkip;
if (displayRightAction()) {
contentWidth -= st::msgPhotoSkip;
}
//} else if (!Adaptive::Wide() && !out() && !fromChannel() && st::msgPhotoSkip - (hmaxwidth - hwidth) > 0) { //} else if (!Adaptive::Wide() && !out() && !fromChannel() && st::msgPhotoSkip - (hmaxwidth - hwidth) > 0) {
// contentLeft += st::msgPhotoSkip - (hmaxwidth - hwidth); // contentLeft += st::msgPhotoSkip - (hmaxwidth - hwidth);
} }
const auto media = this->media();
const auto mediaWidth = media ? media->width() : width();
const auto availableWidth = width() - st::msgMargin.left() - st::msgMargin.right();
auto contentWidth = availableWidth;
if (item->history()->peer->isSelf() && !outbg) {
contentWidth -= st::msgPhotoSkip;
}
accumulate_min(contentWidth, maxWidth()); accumulate_min(contentWidth, maxWidth());
accumulate_min(contentWidth, st::msgMaxWidth); accumulate_min(contentWidth, st::msgMaxWidth);
if (mediaWidth < contentWidth) { if (mediaWidth < contentWidth) {
@ -1546,7 +1550,7 @@ int Message::resizeContentGetHeight(int newWidth) {
// This code duplicates countGeometry() but also resizes media. // This code duplicates countGeometry() but also resizes media.
auto contentWidth = newWidth - (st::msgMargin.left() + st::msgMargin.right()); auto contentWidth = newWidth - (st::msgMargin.left() + st::msgMargin.right());
if (item->history()->peer->isSelf() && !hasOutLayout()) { if (hasFromPhoto() && displayRightAction()) {
contentWidth -= st::msgPhotoSkip; contentWidth -= st::msgPhotoSkip;
} }
accumulate_min(contentWidth, maxWidth()); accumulate_min(contentWidth, maxWidth());

View File

@ -44,8 +44,8 @@ void SectionWidget::init() {
}, _content->lifetime()); }, _content->lifetime());
} }
PeerData *SectionWidget::activePeer() const { Dialogs::RowDescriptor SectionWidget::activeChat() const {
return _content->activePeer(); return _content->activeChat();
} }
bool SectionWidget::hasTopBarShadow() const { bool SectionWidget::hasTopBarShadow() const {

View File

@ -35,7 +35,7 @@ public:
Wrap wrap, Wrap wrap,
not_null<MoveMemento*> memento); not_null<MoveMemento*> memento);
PeerData *activePeer() const override; Dialogs::RowDescriptor activeChat() const override;
bool hasTopBarShadow() const override; bool hasTopBarShadow() const override;
QPixmap grabForShowAnimation( QPixmap grabForShowAnimation(

View File

@ -104,17 +104,25 @@ void WrapWidget::startInjectingActivePeerProfiles() {
using namespace rpl::mappers; using namespace rpl::mappers;
rpl::combine( rpl::combine(
_wrap.value(), _wrap.value(),
_controller->parentController()->activePeer.value() _controller->parentController()->activeChatValue()
) | rpl::filter( ) | rpl::filter(
(_1 == Wrap::Side) && (_2 != nullptr) (_1 == Wrap::Side) && _2
) | rpl::map( ) | rpl::map(
_2 _2
) | rpl::start_with_next([this](not_null<PeerData*> peer) { ) | rpl::start_with_next([this](Dialogs::Key key) {
injectActivePeerProfile(peer); injectActiveProfile(key);
}, lifetime()); }, lifetime());
} }
void WrapWidget::injectActiveProfile(Dialogs::Key key) {
if (const auto peer = key.peer()) {
injectActivePeerProfile(peer);
} else if (const auto feed = key.feed()) {
// #TODO feed profile
}
}
void WrapWidget::injectActivePeerProfile(not_null<PeerData*> peer) { void WrapWidget::injectActivePeerProfile(not_null<PeerData*> peer) {
const auto firstPeerId = hasStackHistory() const auto firstPeerId = hasStackHistory()
? _historyStack.front().section->peerId() ? _historyStack.front().section->peerId()
@ -169,6 +177,10 @@ not_null<PeerData*> WrapWidget::peer() const {
return _controller->peer(); return _controller->peer();
} }
Dialogs::RowDescriptor WrapWidget::activeChat() const {
return Dialogs::RowDescriptor(App::history(peer()), MsgId(0));
}
// This was done for tabs support. // This was done for tabs support.
// //
//void WrapWidget::createTabs() { //void WrapWidget::createTabs() {

View File

@ -78,9 +78,7 @@ public:
not_null<Memento*> memento); not_null<Memento*> memento);
not_null<PeerData*> peer() const; not_null<PeerData*> peer() const;
PeerData *activePeer() const override { Dialogs::RowDescriptor activeChat() const override;
return peer();
}
Wrap wrap() const { Wrap wrap() const {
return _wrap.current(); return _wrap.current();
} }
@ -140,6 +138,7 @@ private:
struct StackItem; struct StackItem;
void startInjectingActivePeerProfiles(); void startInjectingActivePeerProfiles();
void injectActiveProfile(Dialogs::Key key);
void injectActivePeerProfile(not_null<PeerData*> peer); void injectActivePeerProfile(not_null<PeerData*> peer);
void restoreHistoryStack( void restoreHistoryStack(
std::vector<std::unique_ptr<ContentMemento>> stack); std::vector<std::unique_ptr<ContentMemento>> stack);

View File

@ -223,7 +223,6 @@ MainWidget::MainWidget(
connect(&_byPtsTimer, SIGNAL(timeout()), this, SLOT(onGetDifferenceTimeByPts())); connect(&_byPtsTimer, SIGNAL(timeout()), this, SLOT(onGetDifferenceTimeByPts()));
connect(&_byMinChannelTimer, SIGNAL(timeout()), this, SLOT(getDifference())); connect(&_byMinChannelTimer, SIGNAL(timeout()), this, SLOT(getDifference()));
connect(&_failDifferenceTimer, SIGNAL(timeout()), this, SLOT(onGetDifferenceTimeAfterFail())); connect(&_failDifferenceTimer, SIGNAL(timeout()), this, SLOT(onGetDifferenceTimeAfterFail()));
connect(_history, SIGNAL(historyShown(History*,MsgId)), this, SLOT(onHistoryShown(History*,MsgId)));
connect(&updateNotifySettingTimer, SIGNAL(timeout()), this, SLOT(onUpdateNotifySettings())); connect(&updateNotifySettingTimer, SIGNAL(timeout()), this, SLOT(onUpdateNotifySettings()));
subscribe(Media::Player::Updated(), [this](const AudioMsgId &audioId) { subscribe(Media::Player::Updated(), [this](const AudioMsgId &audioId) {
if (audioId.type() != AudioMsgId::Type::Video) { if (audioId.type() != AudioMsgId::Type::Video) {
@ -250,15 +249,16 @@ MainWidget::MainWidget(
}); });
using namespace rpl::mappers; using namespace rpl::mappers;
_controller->activePeer.value( _controller->activeChatValue(
) | rpl::map([](PeerData *peer) { ) | rpl::map([](Dialogs::Key key) {
const auto peer = key.peer();
auto canWrite = peer auto canWrite = peer
? Data::CanWriteValue(peer) ? Data::CanWriteValue(peer)
: rpl::single(false); : rpl::single(false);
return std::move(canWrite) | rpl::map(tuple(peer, _1)); return std::move(canWrite) | rpl::map(tuple(key, _1));
}) | rpl::flatten_latest( }) | rpl::flatten_latest(
) | rpl::start_with_next([this](PeerData *peer, bool canWrite) { ) | rpl::start_with_next([this](Dialogs::Key key, bool canWrite) {
updateThirdColumnToCurrentPeer(peer, canWrite); updateThirdColumnToCurrentChat(key, canWrite);
}, lifetime()); }, lifetime());
QCoreApplication::instance()->installEventFilter(this); QCoreApplication::instance()->installEventFilter(this);
@ -813,7 +813,6 @@ void MainWidget::noHider(HistoryHider *destroyed) {
_forwardConfirm->closeBox(); _forwardConfirm->closeBox();
_forwardConfirm = nullptr; _forwardConfirm = nullptr;
} }
onHistoryShown(_history->history(), _history->msgId());
if (_mainSection || (_history->peer() && _history->peer()->id)) { if (_mainSection || (_history->peer() && _history->peer()->id)) {
auto animationParams = ([this] { auto animationParams = ([this] {
if (_mainSection) { if (_mainSection) {
@ -851,7 +850,6 @@ void MainWidget::hiderLayer(object_ptr<HistoryHider> h) {
_hider->hide(); _hider->hide();
auto animationParams = prepareDialogsAnimation(); auto animationParams = prepareDialogsAnimation();
onHistoryShown(nullptr, 0);
if (_mainSection) { if (_mainSection) {
_mainSection->hide(); _mainSection->hide();
} else { } else {
@ -1039,11 +1037,13 @@ void MainWidget::removeDialog(Dialogs::Key key) {
_dialogs->removeDialog(key); _dialogs->removeDialog(key);
} }
void MainWidget::deleteConversation(PeerData *peer, bool deleteHistory) { void MainWidget::deleteConversation(
if (activePeer() == peer) { not_null<PeerData*> peer,
bool deleteHistory) {
if (_controller->activeChatCurrent().peer() == peer) {
Ui::showChatsList(); Ui::showChatsList();
} }
if (auto history = App::historyLoaded(peer->id)) { if (const auto history = App::historyLoaded(peer->id)) {
Auth().data().setPinnedDialog(history, false); Auth().data().setPinnedDialog(history, false);
removeDialog(history); removeDialog(history);
if (peer->isMegagroup() && peer->asChannel()->mgInfo->migrateFromPtr) { if (peer->isMegagroup() && peer->asChannel()->mgInfo->migrateFromPtr) {
@ -2163,7 +2163,7 @@ void MainWidget::ui_showPeerHistory(
} }
} }
auto wasActivePeer = activePeer(); const auto wasActivePeer = _controller->activeChatCurrent().peer();
if (params.activation != anim::activation::background) { if (params.activation != anim::activation::background) {
Ui::hideSettingsAndLayer(); Ui::hideSettingsAndLayer();
} }
@ -2200,9 +2200,7 @@ void MainWidget::ui_showPeerHistory(
auto animationParams = animatedShow() ? prepareHistoryAnimation(peerId) : Window::SectionSlideParams(); auto animationParams = animatedShow() ? prepareHistoryAnimation(peerId) : Window::SectionSlideParams();
if (back || (way == Way::ClearStack)) { if (!back && (way != Way::ClearStack)) {
clearEntryInStack();
} else {
// This may modify the current section, for example remove its contents. // This may modify the current section, for example remove its contents.
saveSectionInStack(); saveSectionInStack();
} }
@ -2232,14 +2230,17 @@ void MainWidget::ui_showPeerHistory(
} }
} }
} else { } else {
if (!noPeer && wasActivePeer != activePeer()) { const auto nowActivePeer = _controller->activeChatCurrent().peer();
if (activePeer()->isChannel()) { if (nowActivePeer && nowActivePeer != wasActivePeer) {
activePeer()->asChannel()->ptsWaitingForShortPoll( if (const auto channel = nowActivePeer->asChannel()) {
channel->ptsWaitingForShortPoll(
WaitForChannelGetDifference); WaitForChannelGetDifference);
} }
_viewsIncremented.remove(activePeer()); _viewsIncremented.remove(nowActivePeer);
}
if (Adaptive::OneColumn() && !_dialogs->isHidden()) {
_dialogs->hide();
} }
if (Adaptive::OneColumn() && !_dialogs->isHidden()) _dialogs->hide();
if (!_a_show.animating()) { if (!_a_show.animating()) {
if (!animationParams.oldContentCache.isNull()) { if (!animationParams.oldContentCache.isNull()) {
_history->showAnimated( _history->showAnimated(
@ -2255,9 +2256,6 @@ void MainWidget::ui_showPeerHistory(
} }
} }
} }
//if (wasActivePeer && wasActivePeer->isChannel() && activePeer() != wasActivePeer) {
// wasActivePeer->asChannel()->ptsWaitingForShortPoll(false);
//}
if (!_dialogs->isHidden()) { if (!_dialogs->isHidden()) {
if (!back) { if (!back) {
@ -2268,8 +2266,8 @@ void MainWidget::ui_showPeerHistory(
_dialogs->update(); _dialogs->update();
} }
if (!peerId) { if (noPeer) {
_controller->activePeer = nullptr; _controller->setActiveChatEntry(Dialogs::Key());
} }
checkFloatPlayerVisibility(); checkFloatPlayerVisibility();
@ -2299,34 +2297,6 @@ PeerData *MainWidget::peer() {
return _history->peer(); return _history->peer();
} }
PeerData *MainWidget::activePeer() {
if (const auto history = _history->history()) {
return history->peer;
} else if (_historyInStack) {
return _historyInStack->peer;
}
return nullptr;
}
MsgId MainWidget::activeMsgId() {
return _history->peer() ? _history->msgId() : _msgIdInStack;
}
void MainWidget::setEntryInStack(not_null<History*> history, MsgId msgId) {
setEntryInStackValues(history, msgId);
}
void MainWidget::clearEntryInStack() {
setEntryInStackValues(nullptr, 0);
}
void MainWidget::setEntryInStackValues(History *history, MsgId msgId) {
updateCurrentChatListEntry();
_historyInStack = history;
_msgIdInStack = msgId;
updateCurrentChatListEntry();
}
void MainWidget::saveSectionInStack() { void MainWidget::saveSectionInStack() {
if (_mainSection) { if (_mainSection) {
if (auto memento = _mainSection->createMemento()) { if (auto memento = _mainSection->createMemento()) {
@ -2335,10 +2305,9 @@ void MainWidget::saveSectionInStack() {
_stack.back()->setThirdSectionWeak(_thirdSection.data()); _stack.back()->setThirdSectionWeak(_thirdSection.data());
} }
} else if (const auto history = _history->history()) { } else if (const auto history = _history->history()) {
setEntryInStack(history, _history->msgId());
_stack.push_back(std::make_unique<StackItemHistory>( _stack.push_back(std::make_unique<StackItemHistory>(
_historyInStack, history,
_msgIdInStack, _history->msgId(),
_history->replyReturns())); _history->replyReturns()));
_stack.back()->setThirdSectionWeak(_thirdSection.data()); _stack.back()->setThirdSectionWeak(_thirdSection.data());
} }
@ -2618,7 +2587,9 @@ void MainWidget::showNewSection(
} }
if (settingSection.data() == _mainSection.data()) { if (settingSection.data() == _mainSection.data()) {
_controller->activePeer = _mainSection->activePeer(); if (const auto entry = _mainSection->activeChat(); entry.key) {
_controller->setActiveChatEntry(entry);
}
} }
checkFloatPlayerVisibility(); checkFloatPlayerVisibility();
@ -2666,7 +2637,9 @@ void MainWidget::showBackFromStack(
if (selectingPeer()) return; if (selectingPeer()) return;
if (_stack.empty()) { if (_stack.empty()) {
_controller->clearSectionStack(params); _controller->clearSectionStack(params);
if (App::wnd()) QTimer::singleShot(0, App::wnd(), SLOT(setInnerFocus())); if (App::wnd()) {
QTimer::singleShot(0, App::wnd(), SLOT(setInnerFocus()));
}
return; return;
} }
auto item = std::move(_stack.back()); auto item = std::move(_stack.back());
@ -2676,14 +2649,6 @@ void MainWidget::showBackFromStack(
} }
_thirdSectionFromStack = item->takeThirdSectionMemento(); _thirdSectionFromStack = item->takeThirdSectionMemento();
if (item->type() == HistoryStackItem) { if (item->type() == HistoryStackItem) {
clearEntryInStack();
for (auto i = _stack.size(); i > 0;) {
if (_stack[--i]->type() == HistoryStackItem) {
auto historyItem = static_cast<StackItemHistory*>(_stack[i].get());
setEntryInStack(historyItem->history, historyItem->msgId);
break;
}
}
auto historyItem = static_cast<StackItemHistory*>(item.get()); auto historyItem = static_cast<StackItemHistory*>(item.get());
_controller->showPeerHistory( _controller->showPeerHistory(
historyItem->peer()->id, historyItem->peer()->id,
@ -2806,12 +2771,6 @@ QPixmap MainWidget::grabForShowAnimation(const Window::SectionSlideParams &param
return result; return result;
} }
void MainWidget::updateCurrentChatListEntry() {
if (_historyInStack) {
_dialogs->dlgUpdated(_historyInStack, _msgIdInStack);
}
}
void MainWidget::dlgUpdated( void MainWidget::dlgUpdated(
Dialogs::Mode list, Dialogs::Mode list,
not_null<Dialogs::Row*> row) { not_null<Dialogs::Row*> row) {
@ -3248,18 +3207,23 @@ bool MainWidget::saveThirdSectionToStackBack() const {
} }
auto MainWidget::thirdSectionForCurrentMainSection( auto MainWidget::thirdSectionForCurrentMainSection(
not_null<PeerData*> peer) Dialogs::Key key)
-> std::unique_ptr<Window::SectionMemento> { -> std::unique_ptr<Window::SectionMemento> {
if (_thirdSectionFromStack) { if (_thirdSectionFromStack) {
return std::move(_thirdSectionFromStack); return std::move(_thirdSectionFromStack);
} else if (const auto peer = key.peer()) {
return std::make_unique<Info::Memento>(
peer->id,
Info::Memento::DefaultSection(peer));
} else {
return std::make_unique<Info::Memento>(
App::self()->id,
Info::Memento::DefaultSection(App::self()));
} }
return std::make_unique<Info::Memento>(
peer->id,
Info::Memento::DefaultSection(peer));
} }
void MainWidget::updateThirdColumnToCurrentPeer( void MainWidget::updateThirdColumnToCurrentChat(
PeerData *peer, Dialogs::Key key,
bool canWrite) { bool canWrite) {
auto saveOldThirdSection = [&] { auto saveOldThirdSection = [&] {
if (saveThirdSectionToStackBack()) { if (saveThirdSectionToStackBack()) {
@ -3285,7 +3249,7 @@ void MainWidget::updateThirdColumnToCurrentPeer(
} }
_controller->showSection( _controller->showSection(
std::move(*thirdSectionForCurrentMainSection(peer)), std::move(*thirdSectionForCurrentMainSection(key)),
params.withThirdColumn()); params.withThirdColumn());
}; };
auto switchTabbedFast = [&] { auto switchTabbedFast = [&] {
@ -3294,7 +3258,7 @@ void MainWidget::updateThirdColumnToCurrentPeer(
}; };
if (Adaptive::ThreeColumn() if (Adaptive::ThreeColumn()
&& Auth().settings().tabbedSelectorSectionEnabled() && Auth().settings().tabbedSelectorSectionEnabled()
&& peer) { && key) {
if (!canWrite) { if (!canWrite) {
switchInfoFast(); switchInfoFast();
Auth().settings().setTabbedSelectorSectionEnabled(true); Auth().settings().setTabbedSelectorSectionEnabled(true);
@ -3305,7 +3269,7 @@ void MainWidget::updateThirdColumnToCurrentPeer(
} }
} else { } else {
Auth().settings().setTabbedReplacedWithInfo(false); Auth().settings().setTabbedReplacedWithInfo(false);
if (!peer) { if (!key) {
if (_thirdSection) { if (_thirdSection) {
_thirdSection.destroy(); _thirdSection.destroy();
_thirdShadow.destroy(); _thirdShadow.destroy();
@ -3449,13 +3413,6 @@ int MainWidget::backgroundFromY() const {
return -getMainSectionTop(); return -getMainSectionTop();
} }
void MainWidget::onHistoryShown(History *history, MsgId atMsgId) {
// updateControlsGeometry();
if (history) {
dlgUpdated(history, atMsgId);
}
}
void MainWidget::searchInPeer(PeerData *peer) { void MainWidget::searchInPeer(PeerData *peer) {
_dialogs->searchInPeer(peer); _dialogs->searchInPeer(peer);
if (Adaptive::OneColumn()) { if (Adaptive::OneColumn()) {
@ -3626,7 +3583,7 @@ void MainWidget::gotChannelDifference(ChannelData *channel, const MTPupdates_Cha
if (!isFinal) { if (!isFinal) {
MTP_LOG(0, ("getChannelDifference { good - after not final channelDifference was received }%1").arg(cTestMode() ? " TESTMODE" : "")); MTP_LOG(0, ("getChannelDifference { good - after not final channelDifference was received }%1").arg(cTestMode() ? " TESTMODE" : ""));
getChannelDifference(channel); getChannelDifference(channel);
} else if (activePeer() == channel) { } else if (_controller->activeChatCurrent().peer() == channel) {
channel->ptsWaitingForShortPoll(timeout ? (timeout * 1000) : WaitForChannelGetDifference); channel->ptsWaitingForShortPoll(timeout ? (timeout * 1000) : WaitForChannelGetDifference);
} }
} }

View File

@ -134,8 +134,6 @@ public:
const Dialogs::RowDescriptor &which) const; const Dialogs::RowDescriptor &which) const;
PeerData *peer(); PeerData *peer();
PeerData *activePeer();
MsgId activeMsgId();
int backgroundFromY() const; int backgroundFromY() const;
void showSection( void showSection(
@ -200,7 +198,9 @@ public:
const QVector<MTPint> &ids, const QVector<MTPint> &ids,
bool forEveryone); bool forEveryone);
void deletedContact(UserData *user, const MTPcontacts_Link &result); void deletedContact(UserData *user, const MTPcontacts_Link &result);
void deleteConversation(PeerData *peer, bool deleteHistory = true); void deleteConversation(
not_null<PeerData*> peer,
bool deleteHistory = true);
void deleteAndExit(ChatData *chat); void deleteAndExit(ChatData *chat);
void deleteAllFromUser(ChannelData *channel, UserData *from); void deleteAllFromUser(ChannelData *channel, UserData *from);
@ -357,8 +357,6 @@ public slots:
void updateOnline(bool gotOtherOffline = false); void updateOnline(bool gotOtherOffline = false);
void checkIdleFinish(); void checkIdleFinish();
void onHistoryShown(History *history, MsgId atMsgId);
void searchInPeer(PeerData *peer); void searchInPeer(PeerData *peer);
void onUpdateNotifySettings(); void onUpdateNotifySettings();
@ -425,12 +423,12 @@ private:
void updateMediaPlaylistPosition(int x); void updateMediaPlaylistPosition(int x);
void updateControlsGeometry(); void updateControlsGeometry();
void updateDialogsWidthAnimated(); void updateDialogsWidthAnimated();
void updateThirdColumnToCurrentPeer( void updateThirdColumnToCurrentChat(
PeerData *peer, Dialogs::Key key,
bool canWrite); bool canWrite);
[[nodiscard]] bool saveThirdSectionToStackBack() const; [[nodiscard]] bool saveThirdSectionToStackBack() const;
[[nodiscard]] auto thirdSectionForCurrentMainSection( [[nodiscard]] auto thirdSectionForCurrentMainSection(Dialogs::Key key)
not_null<PeerData*> peer) -> std::unique_ptr<Window::SectionMemento>; -> std::unique_ptr<Window::SectionMemento>;
void userIsContactUpdated(not_null<UserData*> user); void userIsContactUpdated(not_null<UserData*> user);
void createPlayer(); void createPlayer();
@ -439,11 +437,6 @@ private:
void closeBothPlayers(); void closeBothPlayers();
void playerHeightUpdated(); void playerHeightUpdated();
void updateCurrentChatListEntry();
void setEntryInStack(not_null<History*> history, MsgId msgId);
void clearEntryInStack();
void setEntryInStackValues(History *history, MsgId msgId);
void setCurrentCall(Calls::Call *call); void setCurrentCall(Calls::Call *call);
void createCallTopBar(); void createCallTopBar();
void destroyCallTopBar(); void destroyCallTopBar();
@ -588,8 +581,6 @@ private:
QPointer<ConfirmBox> _forwardConfirm; // for single column layout QPointer<ConfirmBox> _forwardConfirm; // for single column layout
object_ptr<HistoryHider> _hider = { nullptr }; object_ptr<HistoryHider> _hider = { nullptr };
std::vector<std::unique_ptr<StackItem>> _stack; std::vector<std::unique_ptr<StackItem>> _stack;
History *_historyInStack = nullptr;
MsgId _msgIdInStack = 0;
int _playerHeight = 0; int _playerHeight = 0;
int _callTopBarHeight = 0; int _callTopBarHeight = 0;

View File

@ -26,7 +26,7 @@ public:
>(); >();
return std::move(initial).start( return std::move(initial).start(
[consumer, previous](auto &&value) { [consumer, previous](auto &&value) {
if (auto exists = *previous) { if (auto &exists = *previous) {
auto &existing = *exists; auto &existing = *exists;
auto next = std::make_tuple( auto next = std::make_tuple(
std::move(existing), std::move(existing),

View File

@ -401,6 +401,35 @@ TEST_CASE("basic operators tests", "[rpl::operators]") {
REQUIRE(*sum == "0-11-22-3"); REQUIRE(*sum == "0-11-22-3");
} }
SECTION("combine_previous test") {
auto sum = std::make_shared<std::string>("");
{
rpl::lifetime lifetime;
event_stream<int> a;
a.events(
) | combine_previous(
) | start_with_next([=](int previous, int next) {
*sum += std::to_string(previous) + ' ';
*sum += std::to_string(next) + ' ';
}, lifetime);
a.events(
) | combine_previous(
5
) | start_with_next([=](int previous, int next) {
*sum += std::to_string(10 + previous) + ' ';
*sum += std::to_string(next) + ' ';
}, lifetime);
a.fire(1);
a.fire(2);
a.fire(3);
a.fire(4);
}
REQUIRE(*sum == "15 1 1 2 11 2 2 3 12 3 3 4 13 4 ");
}
SECTION("take test") { SECTION("take test") {
auto sum = std::make_shared<std::string>(""); auto sum = std::make_shared<std::string>("");
{ {

View File

@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#pragma once #pragma once
#include "ui/rp_widget.h" #include "ui/rp_widget.h"
#include "dialogs/dialogs_key.h"
namespace Window { namespace Window {
@ -69,8 +70,8 @@ class SectionWidget : public AbstractSectionWidget {
public: public:
SectionWidget(QWidget *parent, not_null<Window::Controller*> controller); SectionWidget(QWidget *parent, not_null<Window::Controller*> controller);
virtual PeerData *activePeer() const { virtual Dialogs::RowDescriptor activeChat() const {
return nullptr; return {};
} }
// When resizing the widget with top edge moved up or down and we // When resizing the widget with top edge moved up or down and we

View File

@ -36,6 +36,46 @@ Controller::Controller(not_null<MainWindow*> window)
}, lifetime()); }, lifetime());
} }
void Controller::setActiveChatEntry(Dialogs::RowDescriptor row) {
_activeChatEntry = row;
}
void Controller::setActiveChatEntry(Dialogs::Key key) {
setActiveChatEntry({ key, MsgId(0) });
}
Dialogs::RowDescriptor Controller::activeChatEntryCurrent() const {
return _activeChatEntry.current();
}
Dialogs::Key Controller::activeChatCurrent() const {
return activeChatEntryCurrent().key;
}
auto Controller::activeChatEntryChanges() const
-> rpl::producer<Dialogs::RowDescriptor> {
return _activeChatEntry.changes();
}
rpl::producer<Dialogs::Key> Controller::activeChatChanges() const {
return activeChatEntryChanges(
) | rpl::map([](const Dialogs::RowDescriptor &value) {
return value.key;
}) | rpl::distinct_until_changed();
}
auto Controller::activeChatEntryValue() const
-> rpl::producer<Dialogs::RowDescriptor> {
return _activeChatEntry.value();
}
rpl::producer<Dialogs::Key> Controller::activeChatValue() const {
return activeChatEntryValue(
) | rpl::map([](const Dialogs::RowDescriptor &value) {
return value.key;
}) | rpl::distinct_until_changed();
}
void Controller::enableGifPauseReason(GifPauseReason reason) { void Controller::enableGifPauseReason(GifPauseReason reason) {
if (!(_gifPauseReasons & reason)) { if (!(_gifPauseReasons & reason)) {
auto notify = (static_cast<int>(_gifPauseReasons) < static_cast<int>(reason)); auto notify = (static_cast<int>(_gifPauseReasons) < static_cast<int>(reason));

View File

@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include <rpl/variable.h> #include <rpl/variable.h>
#include "base/flags.h" #include "base/flags.h"
#include "dialogs/dialogs_key.h"
class MainWidget; class MainWidget;
class HistoryMessage; class HistoryMessage;
@ -118,8 +119,14 @@ public:
// Also used in the Info::Profile to toggle Send Message button. // Also used in the Info::Profile to toggle Send Message button.
rpl::variable<PeerData*> historyPeer; rpl::variable<PeerData*> historyPeer;
// This is used for auto-switch in third column Info::Profile. void setActiveChatEntry(Dialogs::RowDescriptor row);
rpl::variable<PeerData*> activePeer; void setActiveChatEntry(Dialogs::Key key);
Dialogs::RowDescriptor activeChatEntryCurrent() const;
Dialogs::Key activeChatCurrent() const;
rpl::producer<Dialogs::RowDescriptor> activeChatEntryChanges() const;
rpl::producer<Dialogs::Key> activeChatChanges() const;
rpl::producer<Dialogs::RowDescriptor> activeChatEntryValue() const;
rpl::producer<Dialogs::Key> activeChatValue() const;
void enableGifPauseReason(GifPauseReason reason); void enableGifPauseReason(GifPauseReason reason);
void disableGifPauseReason(GifPauseReason reason); void disableGifPauseReason(GifPauseReason reason);
@ -237,6 +244,7 @@ private:
base::Observable<void> _gifPauseLevelChanged; base::Observable<void> _gifPauseLevelChanged;
base::Observable<void> _floatPlayerAreaUpdated; base::Observable<void> _floatPlayerAreaUpdated;
rpl::variable<Dialogs::RowDescriptor> _activeChatEntry;
base::Variable<bool> _dialogsListFocused = { false }; base::Variable<bool> _dialogsListFocused = { false };
base::Variable<bool> _dialogsListDisplayForced = { false }; base::Variable<bool> _dialogsListDisplayForced = { false };

View File

@ -158,7 +158,7 @@ Filler::Filler(
bool Filler::showInfo() { bool Filler::showInfo() {
if (_source == PeerMenuSource::Profile || _peer->isSelf()) { if (_source == PeerMenuSource::Profile || _peer->isSelf()) {
return false; return false;
} else if (_controller->activePeer.current() != _peer) { } else if (_controller->activeChatCurrent().peer() != _peer) {
return true; return true;
} else if (!Adaptive::ThreeColumn()) { } else if (!Adaptive::ThreeColumn()) {
return true; return true;
@ -626,14 +626,15 @@ base::lambda<void()> DeleteAndLeaveHandler(not_null<PeerData*> peer) {
: st::attentionBoxButton; : st::attentionBoxButton;
auto callback = [peer] { auto callback = [peer] {
Ui::hideLayer(); Ui::hideLayer();
if (App::wnd()->controller()->activePeer.current() == peer) { const auto controller = App::wnd()->controller();
if (controller->activeChatCurrent().peer() == peer) {
Ui::showChatsList(); Ui::showChatsList();
} }
if (peer->isUser()) { if (peer->isUser()) {
App::main()->deleteConversation(peer); App::main()->deleteConversation(peer);
} else if (auto chat = peer->asChat()) { } else if (const auto chat = peer->asChat()) {
App::main()->deleteAndExit(chat); App::main()->deleteAndExit(chat);
} else if (auto channel = peer->asChannel()) { } else if (const auto channel = peer->asChannel()) {
// Don't delete old history by default, // Don't delete old history by default,
// because Android app doesn't. // because Android app doesn't.
// //