/* 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 "dialogs/dialogs_top_bar_suggestion.h" #include "api/api_credits.h" #include "api/api_peer_photo.h" #include "api/api_premium.h" #include "apiwrap.h" #include "base/call_delayed.h" #include "boxes/star_gift_box.h" // ShowStarGiftBox. #include "core/application.h" #include "core/click_handler_types.h" #include "data/data_birthday.h" #include "data/data_changes.h" #include "data/data_session.h" #include "data/data_user.h" #include "dialogs/ui/dialogs_top_bar_suggestion_content.h" #include "history/view/history_view_group_call_bar.h" #include "info/profile/info_profile_values.h" #include "lang/lang_keys.h" #include "main/main_app_config.h" #include "main/main_session.h" #include "settings/settings_credits_graphics.h" #include "settings/settings_premium.h" #include "ui/controls/userpic_button.h" #include "ui/layers/generic_box.h" #include "ui/text/format_values.h" #include "ui/text/text_utilities.h" #include "ui/ui_utility.h" #include "ui/wrap/slide_wrap.h" #include "window/window_controller.h" #include "window/window_session_controller.h" #include "styles/style_boxes.h" #include "styles/style_chat.h" #include "styles/style_chat_helpers.h" #include "styles/style_dialogs.h" namespace Dialogs { namespace { [[nodiscard]] Window::SessionController *FindSessionController( not_null widget) { const auto window = Core::App().findWindow(widget); return window ? window->sessionController() : nullptr; } constexpr auto kSugSetBirthday = "BIRTHDAY_SETUP"_cs; constexpr auto kSugBirthdayContacts = "BIRTHDAY_CONTACTS_TODAY"_cs; constexpr auto kSugPremiumAnnual = "PREMIUM_ANNUAL"_cs; constexpr auto kSugPremiumUpgrade = "PREMIUM_UPGRADE"_cs; constexpr auto kSugPremiumRestore = "PREMIUM_RESTORE"_cs; constexpr auto kSugPremiumGrace = "PREMIUM_GRACE"_cs; constexpr auto kSugSetUserpic = "USERPIC_SETUP"_cs; constexpr auto kSugLowCreditsSubs = "STARS_SUBSCRIPTION_LOW_BALANCE"_cs; } // namespace rpl::producer*> TopBarSuggestionValue( not_null parent, not_null session, rpl::producer outerWrapToggleValue) { return [=, outerWrapToggleValue = rpl::duplicate(outerWrapToggleValue)]( auto consumer) { auto lifetime = rpl::lifetime(); struct Toggle { bool value = false; anim::type type; }; struct State { TopBarSuggestionContent *content = nullptr; Ui::SlideWrap *wrap = nullptr; rpl::variable desiredWrapToggle; rpl::variable outerWrapToggle; rpl::lifetime birthdayLifetime; rpl::lifetime premiumLifetime; rpl::lifetime userpicLifetime; rpl::lifetime giftsLifetime; rpl::lifetime creditsLifetime; std::unique_ptr creditsHistory; }; const auto state = lifetime.make_state(); state->outerWrapToggle = rpl::duplicate(outerWrapToggleValue); const auto ensureWrap = [=] { if (!state->content) { state->content = Ui::CreateChild( parent); rpl::combine( parent->widthValue(), state->content->desiredHeightValue() ) | rpl::start_with_next([=](int width, int height) { state->content->resize(width, height); }, state->content->lifetime()); } if (!state->wrap) { state->wrap = Ui::CreateChild>( parent, object_ptr::fromRaw(state->content)); state->desiredWrapToggle.force_assign( Toggle{ false, anim::type::instant }); } }; const auto processCurrentSuggestion = [=](auto repeat) -> void { ensureWrap(); const auto content = state->content; const auto wrap = state->wrap; using RightIcon = TopBarSuggestionContent::RightIcon; const auto config = &session->appConfig(); if (session->premiumCanBuy() && config->suggestionCurrent(kSugLowCreditsSubs.utf8())) { state->creditsHistory = std::make_unique( session->user(), false, false); const auto show = [=]( const QString &peers, uint64 needed, uint64 whole) { content->setRightIcon(RightIcon::Close); content->setClickedCallback([=] { const auto controller = FindSessionController(parent); if (!controller) { return; } controller->uiShow()->show(Box( Settings::SmallBalanceBox, controller->uiShow(), needed, Settings::SmallBalanceSubscription{ peers }, [=] { config->dismissSuggestion( kSugLowCreditsSubs.utf8()); repeat(repeat); })); }); content->setHideCallback([=] { config->dismissSuggestion(kSugLowCreditsSubs.utf8()); repeat(repeat); }); content->setContent( tr::lng_dialogs_suggestions_credits_sub_low_title( tr::now, lt_count, float64(needed - whole), lt_emoji, Ui::Text::SingleCustomEmoji(Ui::kCreditsCurrency), lt_channels, { peers }, Ui::Text::Bold), tr::lng_dialogs_suggestions_credits_sub_low_about( tr::now, TextWithEntities::Simple), true); state->desiredWrapToggle.force_assign( Toggle{ true, anim::type::normal }); }; session->credits().load(); state->creditsLifetime.destroy(); session->credits().balanceValue() | rpl::start_with_next([=] { state->creditsLifetime.destroy(); state->creditsHistory->requestSubscriptions( Data::CreditsStatusSlice::OffsetToken(), [=](Data::CreditsStatusSlice slice) { state->creditsHistory = nullptr; auto peers = QStringList(); auto credits = uint64(0); for (const auto &entry : slice.subscriptions) { if (entry.barePeerId) { const auto peer = session->data().peer( PeerId(entry.barePeerId)); peers.append(peer->name()); credits += entry.subscription.credits; } } show( peers.join(", "), credits, session->credits().balance().whole()); }, true); }, state->creditsLifetime); return; } else if (session->premiumCanBuy() && config->suggestionCurrent(kSugPremiumGrace.utf8())) { content->setRightIcon(RightIcon::Close); content->setClickedCallback([=] { const auto controller = FindSessionController(parent); if (!controller) { return; } UrlClickHandler::Open( u"https://t.me/premiumbot?start=status"_q, QVariant::fromValue(ClickHandlerContext{ .sessionWindow = base::make_weak(controller), })); }); content->setHideCallback([=] { config->dismissSuggestion(kSugPremiumGrace.utf8()); repeat(repeat); }); content->setContent( tr::lng_dialogs_suggestions_premium_grace_title( tr::now, Ui::Text::Bold), tr::lng_dialogs_suggestions_premium_grace_about( tr::now, TextWithEntities::Simple)); state->desiredWrapToggle.force_assign( Toggle{ true, anim::type::normal }); return; } else if (session->premiumCanBuy() && config->suggestionCurrent(kSugBirthdayContacts.utf8())) { session->data().contactBirthdays( ) | rpl::start_with_next(crl::guard(content, [=]( std::vector users) { if (users.empty()) { repeat(repeat); return; } const auto controller = FindSessionController(parent); if (!controller) { repeat(repeat); return; } const auto isSingle = users.size() == 1; const auto first = session->data().user(users.front()); content->setRightIcon(RightIcon::Close); content->setClickedCallback([=] { if (isSingle) { Ui::ShowStarGiftBox(controller, first); } else { Ui::ChooseStarGiftRecipient(controller); } }); content->setHideCallback([=] { config->dismissSuggestion( kSugBirthdayContacts.utf8()); controller->showToast( tr::lng_dialogs_suggestions_birthday_contact_dismiss( tr::now)); repeat(repeat); }); auto title = isSingle ? tr::lng_dialogs_suggestions_birthday_contact_title( tr::now, lt_text, { first->name() }, Ui::Text::RichLangValue) : tr::lng_dialogs_suggestions_birthday_contacts_title( tr::now, lt_count, users.size(), Ui::Text::RichLangValue); auto text = isSingle ? tr::lng_dialogs_suggestions_birthday_contact_about( tr::now, TextWithEntities::Simple) : tr::lng_dialogs_suggestions_birthday_contacts_about( tr::now, TextWithEntities::Simple); content->setContent(std::move(title), std::move(text)); const auto leftPadding = st::defaultDialogRow.padding.left(); state->giftsLifetime.destroy(); if (!isSingle) { struct UserViews { std::vector inRow; QImage userpics; base::unique_qptr widget; }; const auto s = state->giftsLifetime.template make_state< UserViews>(); s->widget = base::make_unique_q( content); const auto widget = s->widget.get(); content->sizeValue() | rpl::filter_size( ) | rpl::start_with_next([=](const QSize &size) { widget->resize(size); widget->show(); widget->raise(); }, widget->lifetime()); for (const auto &id : users) { if (const auto user = session->data().user(id)) { s->inRow.push_back({ .peer = user }); } } widget->paintRequest() | rpl::start_with_next([=] { auto p = QPainter(widget); const auto regenerate = [&] { if (s->userpics.isNull()) { return true; } for (auto &entry : s->inRow) { if (entry.uniqueKey != entry.peer->userpicUniqueKey( entry.view)) { return true; } } return false; }(); if (regenerate) { const auto &st = st::historyCommentsUserpics; HistoryView::GenerateUserpicsInRow( s->userpics, s->inRow, st, 3); content->setLeftPadding(leftPadding + (users.size() * st.size - st.shift)); } p.drawImage( leftPadding, (widget->height() - (s->userpics.height() / style::DevicePixelRatio())) / 2, s->userpics); }, widget->lifetime()); } else { using Ptr = base::unique_qptr; const auto ptr = state->giftsLifetime.template make_state( base::make_unique_q( content, first, st::uploadUserpicButton)); const auto fake = ptr->get(); fake->setAttribute(Qt::WA_TransparentForMouseEvents); content->sizeValue() | rpl::filter_size( ) | rpl::start_with_next([=](const QSize &s) { fake->raise(); fake->show(); fake->moveToLeft( leftPadding, (s.height() - fake->height()) / 2); }, content->lifetime()); content->setLeftPadding(fake->width() + leftPadding); } state->desiredWrapToggle.force_assign( Toggle{ true, anim::type::normal }); }), state->giftsLifetime); return; } else if (config->suggestionCurrent(kSugSetBirthday.utf8()) && !Data::IsBirthdayToday(session->user()->birthday())) { content->setRightIcon(RightIcon::Close); content->setClickedCallback([=] { const auto controller = FindSessionController(parent); if (!controller) { return; } Core::App().openInternalUrl( u"internal:edit_birthday"_q, QVariant::fromValue(ClickHandlerContext{ .sessionWindow = base::make_weak(controller), })); state->birthdayLifetime = Info::Profile::BirthdayValue( session->user() ) | rpl::map( Data::IsBirthdayTodayValue ) | rpl::flatten_latest( ) | rpl::distinct_until_changed( ) | rpl::start_with_next([=] { repeat(repeat); }); }); content->setHideCallback([=] { config->dismissSuggestion(kSugSetBirthday.utf8()); repeat(repeat); }); content->setContent( tr::lng_dialogs_suggestions_birthday_title( tr::now, Ui::Text::Bold), tr::lng_dialogs_suggestions_birthday_about( tr::now, TextWithEntities::Simple)); state->desiredWrapToggle.force_assign( Toggle{ true, anim::type::normal }); return; } else if (session->premiumPossible() && !session->premium()) { const auto isPremiumAnnual = config->suggestionCurrent( kSugPremiumAnnual.utf8()); const auto isPremiumRestore = !isPremiumAnnual && config->suggestionCurrent(kSugPremiumRestore.utf8()); const auto isPremiumUpgrade = !isPremiumAnnual && !isPremiumRestore && config->suggestionCurrent(kSugPremiumUpgrade.utf8()); const auto set = [=](QString discount) { constexpr auto kMinus = QChar(0x2212); const auto &title = isPremiumAnnual ? tr::lng_dialogs_suggestions_premium_annual_title : isPremiumRestore ? tr::lng_dialogs_suggestions_premium_restore_title : tr::lng_dialogs_suggestions_premium_upgrade_title; const auto &description = isPremiumAnnual ? tr::lng_dialogs_suggestions_premium_annual_about : isPremiumRestore ? tr::lng_dialogs_suggestions_premium_restore_about : tr::lng_dialogs_suggestions_premium_upgrade_about; content->setContent( title( tr::now, lt_text, { discount.replace(kMinus, QChar()) }, Ui::Text::Bold), description(tr::now, TextWithEntities::Simple)); content->setClickedCallback([=] { const auto controller = FindSessionController(parent); if (!controller) { return; } Settings::ShowPremium(controller, "dialogs_hint"); config->dismissSuggestion(isPremiumAnnual ? kSugPremiumAnnual.utf8() : isPremiumRestore ? kSugPremiumRestore.utf8() : kSugPremiumUpgrade.utf8()); repeat(repeat); }); state->desiredWrapToggle.force_assign( Toggle{ true, anim::type::normal }); }; if (isPremiumAnnual || isPremiumRestore || isPremiumUpgrade) { content->setRightIcon(RightIcon::Arrow); const auto api = &session->api().premium(); api->statusTextValue() | rpl::start_with_next([=] { for (const auto &o : api->subscriptionOptions()) { if (o.months == 12) { set(o.discount); state->premiumLifetime.destroy(); return; } } }, state->premiumLifetime); api->reload(); return; } } if (config->suggestionCurrent(kSugSetUserpic.utf8()) && !session->user()->userpicPhotoId()) { const auto controller = FindSessionController(parent); if (!controller) { return; } content->setRightIcon(RightIcon::Close); const auto upload = Ui::CreateChild( content, &controller->window(), Ui::UserpicButton::Role::ChoosePhoto, st::uploadUserpicButton); const auto leftPadding = st::defaultDialogRow.padding.left(); content->sizeValue() | rpl::filter_size( ) | rpl::start_with_next([=](const QSize &s) { upload->raise(); upload->show(); upload->moveToLeft( leftPadding, (s.height() - upload->height()) / 2); }, content->lifetime()); content->setLeftPadding(upload->width() + leftPadding); upload->chosenImages() | rpl::start_with_next([=]( Ui::UserpicButton::ChosenImage &&chosen) { if (chosen.type == Ui::UserpicButton::ChosenType::Set) { session->api().peerPhoto().upload( session->user(), { std::move(chosen.image), chosen.markup.documentId, chosen.markup.colors, }); } }, upload->lifetime()); state->userpicLifetime = session->changes().peerUpdates( session->user(), Data::PeerUpdate::Flag::Photo ) | rpl::start_with_next([=] { if (session->user()->userpicPhotoId()) { repeat(repeat); } }); content->setHideCallback([=] { config->dismissSuggestion(kSugSetUserpic.utf8()); repeat(repeat); }); content->setClickedCallback([=] { const auto syntetic = [=](QEvent::Type type) { Ui::SendSynteticMouseEvent( upload, type, Qt::LeftButton, upload->mapToGlobal(QPoint(0, 0))); }; syntetic(QEvent::MouseMove); syntetic(QEvent::MouseButtonPress); syntetic(QEvent::MouseButtonRelease); }); content->setContent( tr::lng_dialogs_suggestions_userpics_title( tr::now, Ui::Text::Bold), tr::lng_dialogs_suggestions_userpics_about( tr::now, TextWithEntities::Simple)); state->desiredWrapToggle.force_assign( Toggle{ true, anim::type::normal }); return; } state->desiredWrapToggle.force_assign( Toggle{ false, anim::type::normal }); base::call_delayed(st::slideWrapDuration * 2, wrap, [=] { state->content = nullptr; state->wrap = nullptr; consumer.put_next(nullptr); }); }; state->desiredWrapToggle.value() | rpl::combine_previous( ) | rpl::filter([=] { return state->wrap != nullptr; }) | rpl::start_with_next([=](Toggle was, Toggle now) { state->wrap->toggle( state->outerWrapToggle.current() && now.value, (was.value == now.value) ? anim::type::instant : now.type); }, lifetime); state->outerWrapToggle.value() | rpl::combine_previous( ) | rpl::filter([=] { return state->wrap != nullptr; }) | rpl::start_with_next([=](bool was, bool now) { const auto toggle = state->desiredWrapToggle.current(); state->wrap->toggle( toggle.value && now, (was == now) ? toggle.type : anim::type::instant); }, lifetime); session->appConfig().value() | rpl::start_with_next([=] { const auto was = state->wrap; processCurrentSuggestion(processCurrentSuggestion); if (was != state->wrap) { consumer.put_next_copy(state->wrap); } }, lifetime); return lifetime; }; } } // namespace Dialogs