/* This file is part of Telegram Desktop, the official desktop application for the Telegram messaging service. For license and copyright information please follow this link: https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "info/statistics/info_statistics_inner_widget.h" #include "api/api_statistics.h" #include "apiwrap.h" #include "base/call_delayed.h" #include "base/event_filter.h" #include "data/data_peer.h" #include "data/data_session.h" #include "data/data_stories.h" #include "data/data_story.h" #include "history/history_item.h" #include "info/info_controller.h" #include "info/info_memento.h" #include "info/statistics/info_statistics_list_controllers.h" #include "info/statistics/info_statistics_recent_message.h" #include "info/statistics/info_statistics_widget.h" #include "lang/lang_keys.h" #include "lottie/lottie_icon.h" #include "main/main_session.h" #include "settings/settings_common.h" // CreateLottieIcon. #include "statistics/chart_widget.h" #include "statistics/statistics_common.h" #include "statistics/statistics_format_values.h" #include "statistics/widgets/chart_header_widget.h" #include "ui/layers/generic_box.h" #include "ui/rect.h" #include "ui/vertical_list.h" #include "ui/toast/toast.h" #include "ui/widgets/buttons.h" #include "ui/widgets/popup_menu.h" #include "ui/widgets/scroll_area.h" #include "ui/wrap/slide_wrap.h" #include "styles/style_boxes.h" #include "styles/style_menu_icons.h" #include "styles/style_settings.h" #include "styles/style_statistics.h" namespace Info::Statistics { namespace { struct Descriptor final { not_null peer; not_null api; not_null toastParent; }; void AddContextMenu( not_null button, not_null controller, not_null item) { const auto fullId = item->fullId(); const auto contextMenu = button->lifetime() .make_state>(); const auto showMenu = [=] { *contextMenu = base::make_unique_q( button, st::popupMenuWithIcons); const auto go = [=] { const auto &session = controller->parentController(); if (const auto item = session->session().data().message(fullId)) { session->showMessage(item); } }; contextMenu->get()->addAction( tr::lng_context_to_msg(tr::now), crl::guard(controller, go), &st::menuIconShowInChat); contextMenu->get()->popup(QCursor::pos()); }; base::install_event_filter(button, [=](not_null e) { if (e->type() == QEvent::ContextMenu) { showMenu(); return base::EventFilterResult::Cancel; } return base::EventFilterResult::Continue; }); } void ProcessZoom( const Descriptor &d, not_null widget, const QString &zoomToken, Statistic::ChartViewType type) { if (zoomToken.isEmpty()) { return; } widget->zoomRequests( ) | rpl::start_with_next([=](float64 x) { d.api->requestZoom( zoomToken, x ) | rpl::start_with_next_error_done([=]( const Data::StatisticalGraph &graph) { if (graph.chart) { widget->setZoomedChartData(graph.chart, x, type); } else if (!graph.error.isEmpty()) { Ui::Toast::Show(d.toastParent, graph.error); } }, [=](const QString &error) { }, [=] { }, widget->lifetime()); }, widget->lifetime()); } void FillStatistic( not_null content, const Descriptor &descriptor, Data::AnyStatistics &stats, Fn done) { using Type = Statistic::ChartViewType; const auto &padding = st::statisticsChartEntryPadding; const auto &m = st::statisticsLayerMargins; const auto addSkip = [&](not_null c) { Ui::AddSkip(c, padding.bottom()); Ui::AddDivider(c); Ui::AddSkip(c, padding.top()); }; struct State final { Fn done; int pendingCount = 0; }; const auto state = content->lifetime().make_state( State{ std::move(done) }); const auto singlePendingDone = [=] { state->pendingCount--; if (!state->pendingCount && state->done) { base::take(state->done)(); } }; const auto addChart = [&]( Data::StatisticalGraph &graphData, rpl::producer &&title, Statistic::ChartViewType type) { if (graphData.chart) { const auto widget = content->add( object_ptr(content), m); widget->setChartData(graphData.chart, type); ProcessZoom(descriptor, widget, graphData.zoomToken, type); widget->setTitle(std::move(title)); addSkip(content); } else if (!graphData.zoomToken.isEmpty()) { state->pendingCount++; const auto wrap = content->add( object_ptr>( content, object_ptr(content))); wrap->toggle(false, anim::type::instant); const auto widget = wrap->entity()->add( object_ptr(content), m); descriptor.api->requestZoom( graphData.zoomToken, 0 ) | rpl::start_with_next_error_done([=, graphPtr = &graphData]( const Data::StatisticalGraph &graph) mutable { { // Save the loaded async data to cache. // Guarded by content->lifetime(). *graphPtr = graph; } if (graph.chart) { widget->setChartData(graph.chart, type); wrap->toggle(true, anim::type::instant); ProcessZoom(descriptor, widget, graph.zoomToken, type); widget->setTitle(rpl::duplicate(title)); } else if (!graph.error.isEmpty()) { } }, [=](const QString &error) { singlePendingDone(); }, [=] { singlePendingDone(); }, content->lifetime()); addSkip(wrap->entity()); } }; addSkip(content); if (stats.channel) { addChart( stats.channel.memberCountGraph, tr::lng_chart_title_member_count(), Type::Linear); addChart( stats.channel.joinGraph, tr::lng_chart_title_join(), Type::Linear); addChart( stats.channel.muteGraph, tr::lng_chart_title_mute(), Type::Linear); addChart( stats.channel.viewCountByHourGraph, tr::lng_chart_title_view_count_by_hour(), Type::Linear); addChart( stats.channel.viewCountBySourceGraph, tr::lng_chart_title_view_count_by_source(), Type::StackBar); addChart( stats.channel.joinBySourceGraph, tr::lng_chart_title_join_by_source(), Type::StackBar); addChart( stats.channel.languageGraph, tr::lng_chart_title_language(), Type::StackLinear); addChart( stats.channel.messageInteractionGraph, tr::lng_chart_title_message_interaction(), Type::DoubleLinear); addChart( stats.channel.instantViewInteractionGraph, tr::lng_chart_title_instant_view_interaction(), Type::DoubleLinear); addChart( stats.channel.reactionsByEmotionGraph, tr::lng_chart_title_reactions_by_emotion(), Type::Bar); addChart( stats.channel.storyInteractionsGraph, tr::lng_chart_title_story_interactions(), Type::DoubleLinear); addChart( stats.channel.storyReactionsByEmotionGraph, tr::lng_chart_title_story_reactions_by_emotion(), Type::Bar); } else if (stats.supergroup) { addChart( stats.supergroup.memberCountGraph, tr::lng_chart_title_member_count(), Type::Linear); addChart( stats.supergroup.joinGraph, tr::lng_chart_title_group_join(), Type::Linear); addChart( stats.supergroup.joinBySourceGraph, tr::lng_chart_title_group_join_by_source(), Type::StackBar); addChart( stats.supergroup.languageGraph, tr::lng_chart_title_group_language(), Type::StackLinear); addChart( stats.supergroup.messageContentGraph, tr::lng_chart_title_group_message_content(), Type::StackBar); addChart( stats.supergroup.actionGraph, tr::lng_chart_title_group_action(), Type::DoubleLinear); addChart( stats.supergroup.dayGraph, tr::lng_chart_title_group_day(), Type::Linear); addChart( stats.supergroup.weekGraph, tr::lng_chart_title_group_week(), Type::StackLinear); } else { auto &messageOrStory = stats.message ? stats.message : stats.story; if (messageOrStory) { addChart( messageOrStory.messageInteractionGraph, tr::lng_chart_title_message_interaction(), Type::DoubleLinear); addChart( messageOrStory.reactionsByEmotionGraph, tr::lng_chart_title_reactions_by_emotion(), Type::Bar); } } if (!state->pendingCount) { ++state->pendingCount; singlePendingDone(); } } void AddHeader( not_null content, tr::phrase<> text, const Data::AnyStatistics &stats) { const auto startDate = stats.channel ? stats.channel.startDate : stats.supergroup.startDate; const auto endDate = stats.channel ? stats.channel.endDate : stats.supergroup.endDate; const auto header = content->add( object_ptr(content), st::statisticsLayerMargins + st::statisticsChartHeaderPadding); header->resizeToWidth(header->width()); header->setTitle(text(tr::now)); if (!endDate || !startDate) { header->setSubTitle({}); return; } header->setSubTitle(Statistic::LangDayMonthYear(startDate) + ' ' + QChar(8212) + ' ' + Statistic::LangDayMonthYear(endDate)); } void FillOverview( not_null content, const Data::AnyStatistics &stats, bool isChannelStoryStats) { using Value = Data::StatisticalValue; const auto &channel = stats.channel; const auto &supergroup = stats.supergroup; if (!isChannelStoryStats) { Ui::AddSkip(content, st::statisticsLayerOverviewMargins.top()); AddHeader(content, tr::lng_stats_overview_title, stats); Ui::AddSkip(content); } struct Second final { QColor color; QString text; }; const auto parseSecond = [&](const Value &v) -> Second { const auto diff = v.value - v.previousValue; if (!diff || !v.previousValue) { return {}; } constexpr auto kTooMuchDiff = int(1'000'000); const auto diffAbs = std::abs(diff); const auto diffText = diffAbs > kTooMuchDiff ? Lang::FormatCountToShort(std::abs(diff)).string : QString::number(diffAbs); const auto percentage = std::abs(v.growthRatePercentage); const auto precision = (percentage == int(percentage)) ? 0 : 1; return { (diff < 0 ? st::menuIconAttentionColor : st::settingsIconBg2)->c, QString("%1%2 (%3%)") .arg((diff < 0) ? QChar(0x2212) : QChar(0x002B)) .arg(diffText) .arg(QString::number(percentage, 'f', precision)) }; }; const auto diffBetweenHeaders = 0 + st::statisticsOverviewValue.style.font->height - st::statisticsHeaderTitleTextStyle.font->height; const auto container = content->add( object_ptr(content), st::statisticsLayerMargins); const auto addPrimary = [&](const Value &v) { return Ui::CreateChild( container, (v.value >= 0) ? Lang::FormatCountToShort(v.value).string : QString(), st::statisticsOverviewValue); }; const auto addSub = [&]( not_null primary, const Value &v, tr::phrase<> text) { const auto data = parseSecond(v); const auto second = Ui::CreateChild( container, data.text, st::statisticsOverviewSecondValue); second->setTextColorOverride(data.color); const auto sub = Ui::CreateChild( container, text(), st::statisticsOverviewSubtext); sub->setTextColorOverride(st::windowSubTextFg->c); primary->geometryValue( ) | rpl::start_with_next([=](const QRect &g) { const auto &padding = st::statisticsOverviewSecondValuePadding; second->moveToLeft( rect::right(g) + padding.left(), g.y() + padding.top()); sub->moveToLeft( g.x(), st::statisticsChartHeaderHeight - st::statisticsOverviewSubtext.style.font->height + g.y() + diffBetweenHeaders); }, primary->lifetime()); }; const auto isChannel = (!!channel); const auto &messageOrStory = stats.message ? stats.message : stats.story; const auto isMessage = (!!messageOrStory); const auto hasPostReactions = isChannel && (channel.meanReactionCount.value || channel.meanReactionCount.previousValue); const auto topLeftLabel = (isChannelStoryStats && isChannel) ? addPrimary(channel.meanShareCount) : isChannel ? addPrimary(channel.memberCount) : isMessage ? addPrimary({ .value = float64(messageOrStory.views) }) : addPrimary(supergroup.memberCount); const auto topRightLabel = (isChannelStoryStats && isChannel) ? addPrimary(channel.meanStoryShareCount) : isChannel ? Ui::CreateChild( container, QString("%1%").arg(0.01 * std::round(channel.enabledNotificationsPercentage * 100.)), st::statisticsOverviewValue) : isMessage ? addPrimary({ .value = float64(messageOrStory.publicForwards) }) : addPrimary(supergroup.messageCount); const auto bottomLeftLabel = (isChannelStoryStats && isChannel) ? addPrimary(hasPostReactions ? channel.meanReactionCount : channel.meanStoryReactionCount) : isChannel ? addPrimary(channel.meanViewCount) : isMessage ? addPrimary({ .value = float64(messageOrStory.reactions) }) : addPrimary(supergroup.viewerCount); const auto bottomRightLabel = (isChannelStoryStats && isChannel) ? addPrimary(!hasPostReactions ? Value{ .value = -1 } : channel.meanStoryReactionCount) : isChannel ? addPrimary(channel.meanStoryViewCount) : isMessage ? addPrimary({ .value = float64(messageOrStory.privateForwards) }) : addPrimary(supergroup.senderCount); if (isChannelStoryStats && isChannel) { addSub( topLeftLabel, channel.meanShareCount, tr::lng_stats_overview_mean_share_count); addSub( topRightLabel, channel.meanStoryShareCount, tr::lng_stats_overview_mean_story_share_count); addSub( bottomLeftLabel, hasPostReactions ? channel.meanReactionCount : channel.meanStoryReactionCount, hasPostReactions ? tr::lng_stats_overview_mean_reactions_count : tr::lng_stats_overview_mean_story_reactions_count); if (hasPostReactions) { addSub( bottomRightLabel, channel.meanStoryReactionCount, tr::lng_stats_overview_mean_story_reactions_count); } } else if (const auto &s = channel) { addSub( topLeftLabel, s.memberCount, tr::lng_stats_overview_member_count); addSub( topRightLabel, {}, tr::lng_stats_overview_enabled_notifications); addSub( bottomLeftLabel, s.meanViewCount, tr::lng_stats_overview_mean_view_count); addSub( bottomRightLabel, s.meanStoryViewCount, tr::lng_stats_overview_mean_story_view_count); } else if (const auto &s = supergroup) { addSub( topLeftLabel, s.memberCount, tr::lng_manage_peer_members); addSub( topRightLabel, s.messageCount, tr::lng_stats_overview_messages); addSub( bottomLeftLabel, s.viewerCount, tr::lng_stats_overview_group_mean_view_count); addSub( bottomRightLabel, s.senderCount, tr::lng_stats_overview_group_mean_post_count); } else if (const auto &s = messageOrStory) { if (s.views >= 0) { addSub( topLeftLabel, {}, tr::lng_stats_overview_message_views); } if (s.publicForwards >= 0) { addSub( topRightLabel, {}, tr::lng_stats_overview_message_public_shares); } if (s.reactions >= 0) { addSub( bottomLeftLabel, {}, tr::lng_manage_peer_reactions); } if (s.privateForwards >= 0) { addSub( bottomRightLabel, {}, tr::lng_stats_overview_message_private_shares); } } container->showChildren(); container->resize(container->width(), topLeftLabel->height() * 5); container->sizeValue( ) | rpl::start_with_next([=](const QSize &s) { const auto halfWidth = s.width() / 2; { const auto &p = st::statisticsOverviewValuePadding; topLeftLabel->moveToLeft(p.left(), p.top()); } topRightLabel->moveToLeft( topLeftLabel->x() + halfWidth + st::statisticsOverviewRightSkip, topLeftLabel->y()); bottomLeftLabel->moveToLeft( topLeftLabel->x(), topLeftLabel->y() + st::statisticsOverviewMidSkip); bottomRightLabel->moveToLeft( topRightLabel->x(), bottomLeftLabel->y()); }, container->lifetime()); Ui::AddSkip(content, st::statisticsLayerOverviewMargins.bottom()); } } // namespace void FillLoading( not_null container, rpl::producer toggleOn, rpl::producer<> showFinished) { const auto emptyWrap = container->add( object_ptr>( container, object_ptr(container))); emptyWrap->toggleOn(std::move(toggleOn), anim::type::instant); const auto content = emptyWrap->entity(); auto icon = ::Settings::CreateLottieIcon( content, { .name = u"stats"_q, .sizeOverride = Size(st::changePhoneIconSize) }, st::settingsBlockedListIconPadding); ( std::move(showFinished) | rpl::take(1) ) | rpl::start_with_next([animate = std::move(icon.animate)] { animate(anim::repeat::loop); }, icon.widget->lifetime()); content->add(std::move(icon.widget)); content->add( object_ptr>( content, object_ptr( content, tr::lng_stats_loading(), st::changePhoneTitle)), st::changePhoneTitlePadding + st::boxRowPadding); content->add( object_ptr>( content, object_ptr( content, tr::lng_stats_loading_subtext(), st::statisticsLoadingSubtext)), st::changePhoneDescriptionPadding + st::boxRowPadding); Ui::AddSkip(content, st::settingsBlockedListIconPadding.top()); } InnerWidget::InnerWidget( QWidget *parent, not_null controller, not_null peer, FullMsgId contextId, FullStoryId storyId) : VerticalLayout(parent) , _controller(controller) , _peer(peer) , _contextId(contextId) , _storyId(storyId) { } void InnerWidget::load() { const auto inner = this; const auto descriptor = Descriptor{ _peer, lifetime().make_state(_peer->asChannel()), _controller->uiShow()->toastParent(), }; FillLoading( inner, _loaded.events_starting_with(false) | rpl::map(!rpl::mappers::_1), _showFinished.events()); _showFinished.events( ) | rpl::take(1) | rpl::start_with_next([=] { if (!_contextId && !_storyId) { descriptor.api->request( ) | rpl::start_with_done([=] { _state.stats = Data::AnyStatistics{ descriptor.api->channelStats(), descriptor.api->supergroupStats(), }; fill(); }, lifetime()); } else { const auto lifetimeApi = lifetime().make_state(); const auto api = _storyId ? lifetimeApi->make_state( descriptor.peer->asChannel(), _storyId) : lifetimeApi->make_state( descriptor.peer->asChannel(), _contextId); api->request([=](const Data::StoryStatistics &data) { _state.stats = Data::AnyStatistics{ .message = _contextId ? data : Data::StoryStatistics(), .story = _storyId ? data : Data::StoryStatistics(), }; if (_contextId || _storyId) { _state.publicForwardsFirstSlice = api->firstSlice(); } fill(); lifetimeApi->destroy(); }); } }, lifetime()); } void InnerWidget::fill() { const auto wrap = this->add( object_ptr>( this, object_ptr(this))); wrap->toggle(false, anim::type::instant); const auto inner = wrap->entity(); const auto descriptor = Descriptor{ _peer, lifetime().make_state(_peer->asChannel()), _controller->uiShow()->toastParent(), }; const auto finishLoading = [=] { _loaded.fire(true); wrap->toggle(true, anim::type::instant); this->resizeToWidth(width()); this->showChildren(); }; if (_state.stats.message) { if (const auto i = _peer->owner().message(_contextId)) { Ui::AddSkip(inner); const auto preview = inner->add( object_ptr(inner, i, QImage())); AddContextMenu(preview, _controller, i); Ui::AddSkip(inner); Ui::AddDivider(inner); } } else if (_state.stats.story) { if (const auto story = _peer->owner().stories().lookup(_storyId)) { Ui::AddSkip(inner); const auto preview = inner->add( object_ptr(inner, *story, QImage())); preview->setAttribute(Qt::WA_TransparentForMouseEvents); Ui::AddSkip(inner); Ui::AddDivider(inner); } } FillOverview(inner, _state.stats, false); if (_state.stats.channel) { FillOverview(inner, _state.stats, true); } FillStatistic(inner, descriptor, _state.stats, finishLoading); const auto &channel = _state.stats.channel; const auto &supergroup = _state.stats.supergroup; if (channel) { fillRecentPosts(inner); } else if (supergroup) { const auto showPeerInfo = [=](not_null peer) { _showRequests.fire({ .info = peer->id }); }; const auto addSkip = [&](not_null c) { Ui::AddSkip(c); Ui::AddDivider(c); Ui::AddSkip(c); Ui::AddSkip(c); }; if (!supergroup.topSenders.empty()) { AddMembersList( { .topSenders = supergroup.topSenders }, inner, showPeerInfo, descriptor.peer, tr::lng_stats_members_title()); } if (!supergroup.topAdministrators.empty()) { addSkip(inner); AddMembersList( { .topAdministrators = supergroup.topAdministrators }, inner, showPeerInfo, descriptor.peer, tr::lng_stats_admins_title()); } if (!supergroup.topInviters.empty()) { addSkip(inner); AddMembersList( { .topInviters = supergroup.topInviters }, inner, showPeerInfo, descriptor.peer, tr::lng_stats_inviters_title()); } } else if (_state.stats.message || _state.stats.story) { using namespace Data; AddPublicForwards( _state.publicForwardsFirstSlice, inner, [=](RecentPostId id) { _showRequests.fire({ .info = (!id.messageId && !id.storyId) ? id.messageId.peer : PeerId(0), .history = id.messageId, .story = id.storyId, }); }, descriptor.peer, RecentPostId{ .messageId = _contextId, .storyId = _storyId }); } } void InnerWidget::fillRecentPosts(not_null container) { const auto &stats = _state.stats.channel; if (!stats || stats.recentMessageInteractions.empty()) { return; } _messagePreviews.reserve(stats.recentMessageInteractions.size()); const auto wrap = container->add( object_ptr>( container, object_ptr(container))); const auto content = wrap->entity(); AddHeader(content, tr::lng_stats_recent_messages_title, { stats, {} }); Ui::AddSkip(content); const auto addMessage = [=]( not_null messageWrap, HistoryItem *maybeItem, Data::Story *maybeStory, const Data::StatisticsMessageInteractionInfo &info) { const auto button = messageWrap->add( object_ptr( messageWrap, rpl::never(), st::statisticsRecentPostButton)); const auto fullRecentId = Data::RecentPostId{ .messageId = maybeItem ? maybeItem->fullId() : FullMsgId(), .storyId = maybeStory ? maybeStory->fullId() : FullStoryId(), }; auto it = _state.recentPostPreviews.find(fullRecentId); auto cachedPreview = (it != end(_state.recentPostPreviews)) ? base::take(it->second) : QImage(); const auto raw = maybeItem ? Ui::CreateChild( button, maybeItem, std::move(cachedPreview)) : Ui::CreateChild( button, maybeStory, std::move(cachedPreview)); raw->setInfo( info.viewsCount, info.forwardsCount, info.reactionsCount); if (maybeItem) { AddContextMenu(button, _controller, maybeItem); } _messagePreviews.push_back(raw); raw->show(); button->sizeValue( ) | rpl::start_with_next([=](const QSize &s) { if (!s.isNull()) { raw->setGeometry(Rect(s) - st::statisticsRecentPostButton.padding); } }, raw->lifetime()); button->setClickedCallback([=] { _showRequests.fire({ .messageStatistic = fullRecentId.messageId, .storyStatistic = fullRecentId.storyId, }); }); Ui::AddSkip(messageWrap); if (!wrap->toggled()) { wrap->toggle(true, anim::type::normal); } }; const auto buttonWrap = container->add( object_ptr>( container, object_ptr( container, tr::lng_stories_show_more()))); constexpr auto kFirstPage = int(10); constexpr auto kPerPage = int(30); const auto max = int(stats.recentMessageInteractions.size()); if (_state.recentPostsExpanded) { _state.recentPostsExpanded = std::max( _state.recentPostsExpanded - kPerPage, 0); } const auto showMore = [=] { const auto from = _state.recentPostsExpanded; _state.recentPostsExpanded = std::min( max, _state.recentPostsExpanded ? (_state.recentPostsExpanded + kPerPage) : kFirstPage); if (_state.recentPostsExpanded == max) { buttonWrap->toggle(false, anim::type::instant); } for (auto i = from; i < _state.recentPostsExpanded; i++) { const auto &recent = stats.recentMessageInteractions[i]; const auto messageWrap = content->add( object_ptr(content)); auto &data = _peer->owner(); if (recent.messageId) { const auto fullId = FullMsgId(_peer->id, recent.messageId); if (const auto item = data.message(fullId)) { addMessage(messageWrap, item, nullptr, recent); continue; } const auto callback = crl::guard(content, [=] { if (const auto item = _peer->owner().message(fullId)) { addMessage(messageWrap, item, nullptr, recent); content->resizeToWidth(content->width()); } }); _peer->session().api().requestMessageData( _peer, fullId.msg, callback); } else if (recent.storyId) { const auto fullId = FullStoryId{ _peer->id, recent.storyId }; if (const auto story = data.stories().lookup(fullId)) { addMessage(messageWrap, nullptr, *story, recent); continue; } } } container->resizeToWidth(container->width()); }; const auto delay = st::defaultRippleAnimation.hideDuration; buttonWrap->entity()->setClickedCallback([=] { base::call_delayed(delay, crl::guard(container, showMore)); }); showMore(); if (_messagePreviews.empty()) { wrap->toggle(false, anim::type::instant); } } void InnerWidget::saveState(not_null memento) { for (const auto &message : _messagePreviews) { message->saveState(_state); } memento->setState(base::take(_state)); } void InnerWidget::restoreState(not_null memento) { _state = memento->state(); if (_state.stats.channel || _state.stats.supergroup || _state.stats.message || _state.stats.story) { fill(); } else { load(); } Ui::RpWidget::resizeToWidth(width()); } rpl::producer InnerWidget::scrollToRequests() const { return _scrollToRequests.events(); } auto InnerWidget::showRequests() const -> rpl::producer { return _showRequests.events(); } void InnerWidget::showFinished() { _showFinished.fire({}); } } // namespace Info::Statistics