mirror of https://github.com/procxx/kepka.git
Display active feed state in dialogs list.
This commit is contained in:
parent
840b42934b
commit
47ad5ea98a
|
@ -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 ®ion, 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 ®ion, 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 ®ion, 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 ®ion, 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 ®ion, 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 ®ion, 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;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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)));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -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 ¶ms) {
|
QPixmap Widget::grabForShowAnimation(const Window::SectionSlideParams ¶ms) {
|
||||||
if (params.withTopBarShadow) _fixedBarShadow->hide();
|
if (params.withTopBarShadow) _fixedBarShadow->hide();
|
||||||
auto result = Ui::GrabWidget(this);
|
auto result = Ui::GrabWidget(this);
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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()) {
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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());
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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 ¶m
|
||||||
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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),
|
||||||
|
|
|
@ -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>("");
|
||||||
{
|
{
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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));
|
||||||
|
|
|
@ -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 };
|
||||||
|
|
||||||
|
|
|
@ -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.
|
||||||
//
|
//
|
||||||
|
|
Loading…
Reference in New Issue