Open channel comments, not replies.

This commit is contained in:
John Preston 2020-09-03 12:21:31 +04:00
parent 31e1ed216a
commit fb20be3e6c
9 changed files with 126 additions and 23 deletions

View file

@ -139,7 +139,8 @@ bool RepliesList::applyUpdate(
not_null<Viewer*> viewer, not_null<Viewer*> viewer,
const MessageUpdate &update) { const MessageUpdate &update) {
if (update.item->history() != _history if (update.item->history() != _history
|| update.item->replyToTop() != _rootId) { || update.item->replyToTop() != _rootId
|| !IsServerMsgId(update.item->id)) {
return false; return false;
} }
const auto id = update.item->id; const auto id = update.item->id;

View file

@ -21,7 +21,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/view/history_view_service_message.h" #include "history/view/history_view_service_message.h"
#include "history/view/history_view_cursor_state.h" #include "history/view/history_view_cursor_state.h"
#include "history/view/history_view_context_menu.h" #include "history/view/history_view_context_menu.h"
#include "history/view/history_view_replies_section.h"
#include "ui/widgets/popup_menu.h" #include "ui/widgets/popup_menu.h"
#include "ui/image/image.h" #include "ui/image/image.h"
#include "ui/toast/toast.h" #include "ui/toast/toast.h"
@ -1546,16 +1545,14 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
? tr::lng_comments_view ? tr::lng_comments_view
: tr::lng_replies_view; : tr::lng_replies_view;
_menu->addAction(phrase(tr::now, lt_count, item->repliesCount()), [=] { _menu->addAction(phrase(tr::now, lt_count, item->repliesCount()), [=] {
controller->showSection( controller->showRepliesForMessage(_history, itemId.msg);
HistoryView::RepliesMemento(_history, itemId.msg));
}); });
} else if (const auto replyToTop = item->replyToTop()) { } else if (const auto replyToTop = item->replyToTop()) {
const auto &phrase = item->repliesAreComments() const auto &phrase = item->repliesAreComments()
? tr::lng_comments_view_thread ? tr::lng_comments_view_thread
: tr::lng_replies_view_thread; : tr::lng_replies_view_thread;
_menu->addAction(phrase(tr::now), [=] { _menu->addAction(phrase(tr::now), [=] {
controller->showSection( controller->showRepliesForMessage(_history, replyToTop);
HistoryView::RepliesMemento(_history, replyToTop));
}); });
} }
if (item->allowsEdit(base::unixtime::now())) { if (item->allowsEdit(base::unixtime::now())) {

View file

@ -198,6 +198,11 @@ public:
[[nodiscard]] virtual bool repliesAreComments() const { [[nodiscard]] virtual bool repliesAreComments() const {
return false; return false;
} }
[[nodiscard]] virtual FullMsgId commentsItemId() const {
return FullMsgId();
}
virtual void setCommentsItemId(FullMsgId id) {
}
[[nodiscard]] virtual bool needCheck() const; [[nodiscard]] virtual bool needCheck() const;

View file

@ -34,6 +34,8 @@ struct HistoryMessageVia : public RuntimeComponent<HistoryMessageVia, HistoryIte
}; };
struct HistoryMessageViews : public RuntimeComponent<HistoryMessageViews, HistoryItem> { struct HistoryMessageViews : public RuntimeComponent<HistoryMessageViews, HistoryItem> {
static constexpr auto kMaxRecentRepliers = 3;
struct Part { struct Part {
QString text; QString text;
int textWidth = 0; int textWidth = 0;
@ -42,8 +44,8 @@ struct HistoryMessageViews : public RuntimeComponent<HistoryMessageViews, Histor
std::vector<UserId> recentRepliers; std::vector<UserId> recentRepliers;
Part views; Part views;
Part replies; Part replies;
ChannelId repliesChannelId = 0; ChannelId commentsChannelId = 0;
static constexpr auto kMaxRecentRepliers = 3; MsgId commentsRootId = 0;
}; };
struct HistoryMessageSigned : public RuntimeComponent<HistoryMessageSigned, HistoryItem> { struct HistoryMessageSigned : public RuntimeComponent<HistoryMessageSigned, HistoryItem> {

View file

@ -747,10 +747,10 @@ int HistoryMessage::viewsCount() const {
int HistoryMessage::repliesCount() const { int HistoryMessage::repliesCount() const {
if (const auto views = Get<HistoryMessageViews>()) { if (const auto views = Get<HistoryMessageViews>()) {
if (views->repliesChannelId) { if (views->commentsChannelId) {
if (const auto channel = history()->peer->asChannel()) { if (const auto channel = history()->peer->asChannel()) {
const auto linked = channel->linkedChat(); const auto linked = channel->linkedChat();
if (!linked || linked->bareId() != views->repliesChannelId) { if (!linked || linked->bareId() != views->commentsChannelId) {
return 0; return 0;
} }
} else { } else {
@ -764,11 +764,11 @@ int HistoryMessage::repliesCount() const {
bool HistoryMessage::repliesAreComments() const { bool HistoryMessage::repliesAreComments() const {
if (const auto views = Get<HistoryMessageViews>()) { if (const auto views = Get<HistoryMessageViews>()) {
if (!views->repliesChannelId) { if (!views->commentsChannelId) {
return false; return false;
} else if (const auto channel = history()->peer->asChannel()) { } else if (const auto channel = history()->peer->asChannel()) {
const auto linked = channel->linkedChat(); const auto linked = channel->linkedChat();
if (!linked || linked->bareId() != views->repliesChannelId) { if (!linked || linked->bareId() != views->commentsChannelId) {
return false; return false;
} }
} else { } else {
@ -779,6 +779,21 @@ bool HistoryMessage::repliesAreComments() const {
return HistoryItem::repliesAreComments(); return HistoryItem::repliesAreComments();
} }
FullMsgId HistoryMessage::commentsItemId() const {
if (const auto views = Get<HistoryMessageViews>()) {
return FullMsgId(views->commentsChannelId, views->commentsRootId);
}
return FullMsgId();
}
void HistoryMessage::setCommentsItemId(FullMsgId id) {
if (const auto views = Get<HistoryMessageViews>()) {
if (views->commentsChannelId == id.channel) {
views->commentsRootId = id.msg;
}
}
}
bool HistoryMessage::updateDependencyItem() { bool HistoryMessage::updateDependencyItem() {
if (const auto reply = Get<HistoryMessageReply>()) { if (const auto reply = Get<HistoryMessageReply>()) {
const auto documentId = reply->replyToDocumentId; const auto documentId = reply->replyToDocumentId;
@ -1468,7 +1483,7 @@ void HistoryMessage::setReplies(const MTPMessageReplies &data) {
const auto count = data.vreplies().v; const auto count = data.vreplies().v;
const auto channelId = data.vchannel_id().value_or_empty(); const auto channelId = data.vchannel_id().value_or_empty();
const auto countChanged = (views->replies.count != count); const auto countChanged = (views->replies.count != count);
const auto channelChanged = (views->repliesChannelId != channelId); const auto channelChanged = (views->commentsChannelId != channelId);
const auto recentChanged = (views->recentRepliers != repliers); const auto recentChanged = (views->recentRepliers != repliers);
if (!countChanged && !channelChanged && !recentChanged) { if (!countChanged && !channelChanged && !recentChanged) {
return; return;
@ -1477,7 +1492,7 @@ void HistoryMessage::setReplies(const MTPMessageReplies &data) {
if (recentChanged) { if (recentChanged) {
views->recentRepliers = repliers; views->recentRepliers = repliers;
} }
views->repliesChannelId = channelId; views->commentsChannelId = channelId;
refreshRepliesText(views, channelChanged); refreshRepliesText(views, channelChanged);
}); });
} }
@ -1486,7 +1501,7 @@ void HistoryMessage::refreshRepliesText(
not_null<HistoryMessageViews*> views, not_null<HistoryMessageViews*> views,
bool forceResize) { bool forceResize) {
const auto was = views->replies.textWidth; const auto was = views->replies.textWidth;
if (views->repliesChannelId) { if (views->commentsChannelId) {
views->replies.text = (views->replies.count > 0) views->replies.text = (views->replies.count > 0)
? tr::lng_comments_open_count( ? tr::lng_comments_open_count(
tr::now, tr::now,
@ -1517,7 +1532,7 @@ void HistoryMessage::changeRepliesCount(int delta, UserId replier) {
return; return;
} }
views->replies.count = std::max(views->replies.count + delta, 0); views->replies.count = std::max(views->replies.count + delta, 0);
if (replier && views->repliesChannelId) { if (replier && views->commentsChannelId) {
if (delta < 0) { if (delta < 0) {
views->recentRepliers.erase( views->recentRepliers.erase(
ranges::remove(views->recentRepliers, replier), ranges::remove(views->recentRepliers, replier),

View file

@ -168,6 +168,8 @@ public:
[[nodiscard]] int viewsCount() const override; [[nodiscard]] int viewsCount() const override;
[[nodiscard]] int repliesCount() const override; [[nodiscard]] int repliesCount() const override;
[[nodiscard]] bool repliesAreComments() const override; [[nodiscard]] bool repliesAreComments() const override;
[[nodiscard]] FullMsgId commentsItemId() const override;
void setCommentsItemId(FullMsgId id) override;
bool updateDependencyItem() override; bool updateDependencyItem() override;
[[nodiscard]] MsgId dependencyMsgId() const override { [[nodiscard]] MsgId dependencyMsgId() const override {
return replyToId(); return replyToId();

View file

@ -12,7 +12,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history_message.h" #include "history/history_message.h"
#include "history/view/media/history_view_media.h" #include "history/view/media/history_view_media.h"
#include "history/view/media/history_view_web_page.h" #include "history/view/media/history_view_web_page.h"
#include "history/view/history_view_replies_section.h"
#include "history/history.h" #include "history/history.h"
#include "ui/effects/ripple_animation.h" #include "ui/effects/ripple_animation.h"
#include "core/application.h" #include "core/application.h"
@ -1191,8 +1190,7 @@ bool Message::getStateCommentsButton(
if (const auto window = App::wnd()) { if (const auto window = App::wnd()) {
if (const auto controller = window->sessionController()) { if (const auto controller = window->sessionController()) {
if (const auto item = controller->session().data().message(fullId)) { if (const auto item = controller->session().data().message(fullId)) {
controller->showSection( controller->showRepliesForMessage(item->history(), item->id);
HistoryView::RepliesMemento(item->history(), item->id));
} }
} }
} }
@ -1608,7 +1606,7 @@ void Message::drawInfo(
auto left = infoRight - infoW; auto left = infoRight - infoW;
const auto iconTop = infoBottom + st::historyViewsTop; const auto iconTop = infoBottom + st::historyViewsTop;
const auto textTop = infoBottom - st::msgDateFont->descent; const auto textTop = infoBottom - st::msgDateFont->descent;
if (views->replies.count > 0 && !views->repliesChannelId) { if (views->replies.count > 0 && !views->commentsChannelId) {
auto icon = [&] { auto icon = [&] {
if (item->id > 0) { if (item->id > 0) {
if (outbg) { if (outbg) {
@ -1725,7 +1723,7 @@ int Message::infoWidth() const {
+ views->views.textWidth + views->views.textWidth
+ st::historyViewsWidth; + st::historyViewsWidth;
} }
if (views->replies.count > 0 && !views->repliesChannelId) { if (views->replies.count > 0 && !views->commentsChannelId) {
result += st::historyViewsSpace result += st::historyViewsSpace
+ views->replies.textWidth + views->replies.textWidth
+ st::historyViewsWidth; + st::historyViewsWidth;
@ -1771,7 +1769,7 @@ int Message::timeLeft() const {
if (views->views.count >= 0) { if (views->views.count >= 0) {
result += st::historyViewsSpace + views->views.textWidth + st::historyViewsWidth; result += st::historyViewsSpace + views->views.textWidth + st::historyViewsWidth;
} }
if (views->replies.count > 0 && !views->repliesChannelId) { if (views->replies.count > 0 && !views->commentsChannelId) {
result += st::historyViewsSpace + views->replies.textWidth + st::historyViewsWidth; result += st::historyViewsSpace + views->replies.textWidth + st::historyViewsWidth;
} }
} else if (item->id < 0 && item->history()->peer->isSelf()) { } else if (item->id < 0 && item->history()->peer->isSelf()) {

View file

@ -16,6 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history.h" #include "history/history.h"
#include "history/history_item.h" #include "history/history_item.h"
#include "history/view/history_view_element.h" #include "history/view/history_view_element.h"
#include "history/view/history_view_replies_section.h"
//#include "history/feed/history_feed_section.h" // #feed //#include "history/feed/history_feed_section.h" // #feed
#include "media/player/media_player_instance.h" #include "media/player/media_player_instance.h"
#include "data/data_media_types.h" #include "data/data_media_types.h"
@ -82,10 +83,82 @@ SessionNavigation::SessionNavigation(not_null<Main::Session*> session)
: _session(session) { : _session(session) {
} }
SessionNavigation::~SessionNavigation() {
_session->api().request(base::take(_showingRepliesRequestId)).cancel();
}
Main::Session &SessionNavigation::session() const { Main::Session &SessionNavigation::session() const {
return *_session; return *_session;
} }
void SessionNavigation::showRepliesForMessage(
not_null<History*> history,
MsgId rootId,
const SectionShow &params) {
if (_showingRepliesRequestId
&& _showingRepliesHistory == history.get()
&& _showingRepliesRootId == rootId) {
return;
}
_session->api().request(base::take(_showingRepliesRequestId)).cancel();
const auto channelId = history->channelId();
const auto item = _session->data().message(channelId, rootId);
if (!item || !item->repliesAreComments()) {
if (item->repliesCount() > 0) {
showSection(HistoryView::RepliesMemento(history, rootId));
}
return;
} else if (const auto id = item->commentsItemId()) {
if (const auto item = _session->data().message(id)) {
showSection(
HistoryView::RepliesMemento(item->history(), item->id));
return;
}
}
_showingRepliesHistory = history;
_showingRepliesRootId = rootId;
_showingRepliesRequestId = _session->api().request(
MTPmessages_GetDiscussionMessage(
history->peer->input,
MTP_int(rootId))
).done([=](const MTPmessages_DiscussionMessage &result) {
_showingRepliesRequestId = 0;
result.match([&](const MTPDmessages_discussionMessage &data) {
_session->data().processUsers(data.vusers());
_session->data().processChats(data.vchats());
_session->data().processMessages(
data.vmessages(),
NewMessageType::Existing);
const auto list = data.vmessages().v;
if (list.isEmpty()) {
return;
}
const auto id = IdFromMessage(list.front());
const auto peer = PeerFromMessage(list.front());
if (!peer || !id) {
return;
}
auto item = _session->data().message(
peerToChannel(peer),
id);
if (const auto group = _session->data().groups().find(item)) {
item = group->items.front();
}
if (item) {
const auto post = _session->data().message(channelId, rootId);
if (post) {
post->setCommentsItemId(item->fullId());
}
showSection(
HistoryView::RepliesMemento(item->history(), item->id));
}
});
}).fail([=](const RPCError &error) {
_showingRepliesRequestId = 0;
}).send();
}
void SessionNavigation::showPeerInfo( void SessionNavigation::showPeerInfo(
PeerId peerId, PeerId peerId,
const SectionShow &params) { const SectionShow &params) {

View file

@ -127,6 +127,7 @@ class SessionController;
class SessionNavigation : public base::has_weak_ptr { class SessionNavigation : public base::has_weak_ptr {
public: public:
explicit SessionNavigation(not_null<Main::Session*> session); explicit SessionNavigation(not_null<Main::Session*> session);
virtual ~SessionNavigation();
Main::Session &session() const; Main::Session &session() const;
@ -137,6 +138,11 @@ public:
const SectionShow &params = SectionShow()) = 0; const SectionShow &params = SectionShow()) = 0;
virtual not_null<SessionController*> parentController() = 0; virtual not_null<SessionController*> parentController() = 0;
void showRepliesForMessage(
not_null<History*> history,
MsgId rootId,
const SectionShow &params = SectionShow());
void showPeerInfo( void showPeerInfo(
PeerId peerId, PeerId peerId,
const SectionShow &params = SectionShow()); const SectionShow &params = SectionShow());
@ -157,11 +163,15 @@ public:
FullMsgId contextId, FullMsgId contextId,
const SectionShow &params = SectionShow()); const SectionShow &params = SectionShow());
virtual ~SessionNavigation() = default;
private: private:
const not_null<Main::Session*> _session; const not_null<Main::Session*> _session;
History *_showingRepliesHistory = nullptr;
MsgId _showingRepliesRootId = 0;
mtpRequestId _showingRepliesRequestId = 0;
}; };
class SessionController : public SessionNavigation, private base::Subscriber { class SessionController : public SessionNavigation, private base::Subscriber {