/* 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 "boxes/gift_premium_box.h" #include "api/api_premium.h" #include "api/api_premium_option.h" #include "apiwrap.h" #include "base/unixtime.h" #include "base/weak_ptr.h" #include "boxes/peer_list_controllers.h" // ContactsBoxController. #include "boxes/peers/prepare_short_info_box.h" #include "boxes/peers/replace_boost_box.h" // BoostsForGift. #include "boxes/premium_preview_box.h" // ShowPremiumPreviewBox. #include "core/ui_integration.h" // Core::MarkedTextContext. #include "data/data_boosts.h" #include "data/data_changes.h" #include "data/data_channel.h" #include "data/data_media_types.h" // Data::GiveawayStart. #include "data/data_peer_values.h" // Data::PeerPremiumValue. #include "data/data_session.h" #include "data/data_subscription_option.h" #include "data/data_user.h" #include "data/stickers/data_custom_emoji.h" #include "info/channel_statistics/boosts/giveaway/boost_badge.h" // InfiniteRadialAnimationWidget. #include "lang/lang_keys.h" #include "main/main_session.h" #include "mainwidget.h" #include "payments/payments_checkout_process.h" #include "payments/payments_form.h" #include "settings/settings_premium.h" #include "ui/basic_click_handlers.h" // UrlClickHandler::Open. #include "ui/boxes/boost_box.h" // StartFireworks. #include "ui/controls/userpic_button.h" #include "ui/effects/premium_graphics.h" #include "ui/effects/premium_stars_colored.h" #include "ui/effects/premium_top_bar.h" #include "ui/effects/spoiler_mess.h" #include "ui/layers/generic_box.h" #include "ui/painter.h" #include "ui/rect.h" #include "ui/vertical_list.h" #include "ui/text/text_utilities.h" #include "ui/widgets/checkbox.h" #include "ui/widgets/gradient_round_button.h" #include "ui/wrap/padding_wrap.h" #include "ui/wrap/slide_wrap.h" #include "ui/wrap/table_layout.h" #include "window/window_peer_menu.h" // ShowChooseRecipientBox. #include "window/window_session_controller.h" #include "styles/style_boxes.h" #include "styles/style_giveaway.h" #include "styles/style_info.h" #include "styles/style_layers.h" #include "styles/style_premium.h" #include namespace { constexpr auto kUserpicsMax = size_t(3); using GiftOption = Data::SubscriptionOption; using GiftOptions = Data::SubscriptionOptions; GiftOptions GiftOptionFromTL(const MTPDuserFull &data) { auto result = GiftOptions(); const auto gifts = data.vpremium_gifts(); if (!gifts) { return result; } result = Api::SubscriptionOptionsFromTL(gifts->v); for (auto &option : result) { option.costPerMonth = tr::lng_premium_gift_per( tr::now, lt_cost, option.costPerMonth); } return result; } [[nodiscard]] Fn BoostsForGiftText( const std::vector> users) { Expects(!users.empty()); const auto session = &users.front()->session(); const auto emoji = Ui::Text::SingleCustomEmoji( session->data().customEmojiManager().registerInternalEmoji( st::premiumGiftsBoostIcon, QMargins(0, st::premiumGiftsUserpicBadgeInner, 0, 0), false)); return [=, count = users.size()](TextWithEntities text) { text.append('\n'); text.append('\n'); text.append(tr::lng_premium_gifts_about_reward( tr::now, lt_count, count * BoostsForGift(session), lt_emoji, emoji, Ui::Text::RichLangValue)); return text; }; } using TagUser1 = lngtag_user; using TagUser2 = lngtag_second_user; using TagUser3 = lngtag_name; [[nodiscard]] rpl::producer ComplexAboutLabel( const std::vector> &users, tr::phrase phrase1, tr::phrase phrase2, tr::phrase phrase3, tr::phrase phraseMore) { Expects(!users.empty()); const auto count = users.size(); const auto nameValue = [&](not_null user) { return user->session().changes().peerFlagsValue( user, Data::PeerUpdate::Flag::Name ) | rpl::map([=] { return TextWithEntities{ user->firstName }; }); }; if (count == 1) { return phrase1( lt_user, nameValue(users.front()), Ui::Text::RichLangValue); } else if (count == 2) { return phrase2( lt_user, nameValue(users.front()), lt_second_user, nameValue(users[1]), Ui::Text::RichLangValue); } else if (count == 3) { return phrase3( lt_user, nameValue(users.front()), lt_second_user, nameValue(users[1]), lt_name, nameValue(users[2]), Ui::Text::RichLangValue); } else { return phraseMore( lt_count, rpl::single(count - kUserpicsMax) | tr::to_count(), lt_user, nameValue(users.front()), lt_second_user, nameValue(users[1]), lt_name, nameValue(users[2]), Ui::Text::RichLangValue); } } [[nodiscard]] not_null CircleBadge( not_null parent, const QString &text) { const auto widget = Ui::CreateChild(parent.get()); const auto full = Rect(st::premiumGiftsUserpicBadgeSize); const auto inner = full - Margins(st::premiumGiftsUserpicBadgeInner); auto gradient = QLinearGradient( QPointF(0, full.height()), QPointF(full.width(), 0)); gradient.setStops(Ui::Premium::GiftGradientStops()); widget->paintRequest( ) | rpl::start_with_next([=] { auto p = QPainter(widget); auto hq = PainterHighQualityEnabler(p); p.setPen(Qt::NoPen); p.setBrush(st::boxBg); p.drawEllipse(full); p.setPen(Qt::NoPen); p.setBrush(gradient); p.drawEllipse(inner); p.setFont(st::premiumGiftsUserpicBadgeFont); p.setPen(st::premiumButtonFg); p.drawText(full, text, style::al_center); }, widget->lifetime()); widget->resize(full.size()); return widget; } [[nodiscard]] not_null UserpicsContainer( not_null parent, std::vector> users) { Expects(!users.empty()); if (users.size() == 1) { const auto userpic = Ui::CreateChild( parent.get(), users.front(), st::defaultUserpicButton); userpic->setAttribute(Qt::WA_TransparentForMouseEvents); return userpic; } const auto &singleSize = st::defaultUserpicButton.size; const auto container = Ui::CreateChild(parent.get()); const auto single = singleSize.width(); const auto shift = single - st::boostReplaceUserpicsShift; const auto maxWidth = users.size() * (single - shift) + shift; container->resize(maxWidth, singleSize.height()); container->setAttribute(Qt::WA_TransparentForMouseEvents); const auto diff = (single - st::premiumGiftsUserpicButton.size.width()) / 2; for (auto i = 0; i < users.size(); i++) { const auto bg = Ui::CreateChild(container); bg->resize(singleSize); bg->paintRequest( ) | rpl::start_with_next([=] { auto p = QPainter(bg); auto hq = PainterHighQualityEnabler(p); p.setPen(Qt::NoPen); p.setBrush(st::boxBg); p.drawEllipse(bg->rect()); }, bg->lifetime()); bg->moveToLeft(std::max(0, i * (single - shift)), 0); const auto userpic = Ui::CreateChild( bg, users[i], st::premiumGiftsUserpicButton); userpic->moveToLeft(diff, diff); } return container; } void GiftBox( not_null box, not_null controller, not_null user, GiftOptions options) { const auto boxWidth = st::boxWideWidth; box->setWidth(boxWidth); box->setNoContentMargin(true); const auto buttonsParent = box->verticalLayout().get(); struct State { rpl::event_stream buttonText; }; const auto state = box->lifetime().make_state(); const auto userpicPadding = st::premiumGiftUserpicPadding; const auto top = box->addRow(object_ptr( buttonsParent, userpicPadding.top() + userpicPadding.bottom() + st::defaultUserpicButton.size.height())); using ColoredMiniStars = Ui::Premium::ColoredMiniStars; const auto stars = box->lifetime().make_state( top, true); const auto userpic = UserpicsContainer(top, { user }); userpic->setAttribute(Qt::WA_TransparentForMouseEvents); top->widthValue( ) | rpl::start_with_next([=](int width) { userpic->moveToLeft( (width - userpic->width()) / 2, userpicPadding.top()); const auto center = top->rect().center(); const auto size = QSize( userpic->width() * Ui::Premium::MiniStars::kSizeFactor, userpic->height()); const auto ministarsRect = QRect( QPoint(center.x() - size.width(), center.y() - size.height()), QPoint(center.x() + size.width(), center.y() + size.height())); stars->setPosition(ministarsRect.topLeft()); stars->setSize(ministarsRect.size()); }, userpic->lifetime()); top->paintRequest( ) | rpl::start_with_next([=](const QRect &r) { auto p = QPainter(top); p.fillRect(r, Qt::transparent); stars->paint(p); }, top->lifetime()); const auto close = Ui::CreateChild( buttonsParent, st::infoTopBarClose); close->setClickedCallback([=] { box->closeBox(); }); buttonsParent->widthValue( ) | rpl::start_with_next([=](int width) { close->moveToRight(0, 0, width); }, close->lifetime()); // Header. const auto &padding = st::premiumGiftAboutPadding; const auto available = boxWidth - padding.left() - padding.right(); const auto &stTitle = st::premiumPreviewAboutTitle; auto titleLabel = object_ptr( box, tr::lng_premium_gift_title(), stTitle); titleLabel->resizeToWidth(available); box->addRow( object_ptr>( box, std::move(titleLabel)), st::premiumGiftTitlePadding); auto textLabel = object_ptr(box, st::premiumPreviewAbout); tr::lng_premium_gift_about( lt_user, user->session().changes().peerFlagsValue( user, Data::PeerUpdate::Flag::Name ) | rpl::map([=] { return TextWithEntities{ user->firstName }; }), Ui::Text::RichLangValue ) | rpl::map( BoostsForGiftText({ user }) ) | rpl::start_with_next([ raw = textLabel.data(), session = &user->session()](const TextWithEntities &t) { raw->setMarkedText(t, Core::MarkedTextContext{ .session = session }); }, textLabel->lifetime()); textLabel->setTextColorOverride(stTitle.textFg->c); textLabel->resizeToWidth(available); box->addRow( object_ptr>(box, std::move(textLabel)), padding); // List. const auto group = std::make_shared(); const auto groupValueChangedCallback = [=](int value) { Expects(value < options.size() && value >= 0); auto text = tr::lng_premium_gift_button( tr::now, lt_cost, options[value].costTotal); state->buttonText.fire(std::move(text)); }; group->setChangedCallback(groupValueChangedCallback); Ui::Premium::AddGiftOptions( buttonsParent, group, options, st::premiumGiftOption); // Footer. auto terms = object_ptr( box, tr::lng_premium_gift_terms( lt_link, tr::lng_premium_gift_terms_link( ) | rpl::map([=](const QString &t) { return Ui::Text::Link(t, 1); }), Ui::Text::WithEntities), st::premiumGiftTerms); terms->setLink(1, std::make_shared([=] { box->closeBox(); Settings::ShowPremium(&user->session(), QString()); })); terms->resizeToWidth(available); box->addRow( object_ptr>(box, std::move(terms)), st::premiumGiftTermsPadding); // Button. const auto &stButton = st::premiumGiftBox; box->setStyle(stButton); auto raw = Settings::CreateSubscribeButton({ controller, box, [] { return u"gift"_q; }, state->buttonText.events(), Ui::Premium::GiftGradientStops(), [=] { const auto value = group->current(); return (value < options.size() && value >= 0) ? options[value].botUrl : QString(); }, }); auto button = object_ptr::fromRaw(raw); button->resizeToWidth(boxWidth - rect::m::sum::h(stButton.buttonPadding)); box->setShowFinishedCallback([raw = button.data()] { raw->startGlareAnimation(); }); box->addButton(std::move(button)); groupValueChangedCallback(0); Data::PeerPremiumValue( user ) | rpl::skip(1) | rpl::start_with_next([=] { box->closeBox(); }, box->lifetime()); } void GiftsBox( not_null box, not_null controller, std::vector> users, not_null api, const QString &ref) { Expects(!users.empty()); const auto boxWidth = st::boxWideWidth; box->setWidth(boxWidth); box->setNoContentMargin(true); const auto buttonsParent = box->verticalLayout().get(); const auto session = &users.front()->session(); struct State { rpl::event_stream buttonText; rpl::variable confirmButtonBusy = false; rpl::variable isPaymentComplete = false; }; const auto state = box->lifetime().make_state(); const auto userpicPadding = st::premiumGiftUserpicPadding; const auto top = box->addRow(object_ptr( buttonsParent, userpicPadding.top() + userpicPadding.bottom() + st::defaultUserpicButton.size.height())); using ColoredMiniStars = Ui::Premium::ColoredMiniStars; const auto stars = box->lifetime().make_state( top, true); const auto maxWithUserpic = std::min(users.size(), kUserpicsMax); const auto userpics = UserpicsContainer( top, { users.begin(), users.begin() + maxWithUserpic }); top->widthValue( ) | rpl::start_with_next([=](int width) { userpics->moveToLeft( (width - userpics->width()) / 2, userpicPadding.top()); const auto center = top->rect().center(); const auto size = QSize( userpics->width() * Ui::Premium::MiniStars::kSizeFactor, userpics->height()); const auto ministarsRect = QRect( QPoint(center.x() - size.width(), center.y() - size.height()), QPoint(center.x() + size.width(), center.y() + size.height())); stars->setPosition(ministarsRect.topLeft()); stars->setSize(ministarsRect.size()); }, userpics->lifetime()); if (const auto rest = users.size() - maxWithUserpic; rest > 0) { const auto badge = CircleBadge( userpics, QChar('+') + QString::number(rest)); badge->moveToRight(0, userpics->height() - badge->height()); } top->paintRequest( ) | rpl::start_with_next([=](const QRect &r) { auto p = QPainter(top); p.fillRect(r, Qt::transparent); stars->paint(p); }, top->lifetime()); const auto close = Ui::CreateChild( buttonsParent, st::infoTopBarClose); close->setClickedCallback([=] { box->closeBox(); }); buttonsParent->widthValue( ) | rpl::start_with_next([=](int width) { close->moveToRight(0, 0, width); }, close->lifetime()); // Header. const auto &padding = st::premiumGiftAboutPadding; const auto available = boxWidth - padding.left() - padding.right(); const auto &stTitle = st::premiumPreviewAboutTitle; auto titleLabel = object_ptr( box, rpl::conditional( state->isPaymentComplete.value(), tr::lng_premium_gifts_about_paid_title(), tr::lng_premium_gift_title()), stTitle); titleLabel->resizeToWidth(available); box->addRow( object_ptr>( box, std::move(titleLabel)), st::premiumGiftTitlePadding); // About. { auto text = rpl::conditional( state->isPaymentComplete.value(), ComplexAboutLabel( users, tr::lng_premium_gifts_about_paid1, tr::lng_premium_gifts_about_paid2, tr::lng_premium_gifts_about_paid3, tr::lng_premium_gifts_about_paid_more ) | rpl::map([count = users.size()](TextWithEntities text) { text.append('\n'); text.append('\n'); text.append(tr::lng_premium_gifts_about_paid_below( tr::now, lt_count, float64(count), Ui::Text::RichLangValue)); return text; }), ComplexAboutLabel( users, tr::lng_premium_gifts_about_user1, tr::lng_premium_gifts_about_user2, tr::lng_premium_gifts_about_user3, tr::lng_premium_gifts_about_user_more ) | rpl::map(BoostsForGiftText(users)) ); const auto label = box->addRow( object_ptr>( box, object_ptr(box, st::premiumPreviewAbout)), padding)->entity(); std::move( text ) | rpl::start_with_next([=](const TextWithEntities &t) { using namespace Core; label->setMarkedText(t, MarkedTextContext{ .session = session }); }, label->lifetime()); label->setTextColorOverride(stTitle.textFg->c); label->resizeToWidth(available); } // List. const auto optionsContainer = buttonsParent->add( object_ptr>( buttonsParent, object_ptr(buttonsParent))); const auto options = api->options(users.size()); const auto group = std::make_shared(); const auto groupValueChangedCallback = [=](int value) { Expects(value < options.size() && value >= 0); auto text = tr::lng_premium_gift_button( tr::now, lt_cost, options[value].costTotal); state->buttonText.fire(std::move(text)); }; group->setChangedCallback(groupValueChangedCallback); Ui::Premium::AddGiftOptions( optionsContainer->entity(), group, options, st::premiumGiftOption); optionsContainer->toggleOn( state->isPaymentComplete.value() | rpl::map(!rpl::mappers::_1), anim::type::instant); // Summary. { { // Will be hidden after payment. const auto content = optionsContainer->entity(); Ui::AddSkip(content); Ui::AddDivider(content); Ui::AddSkip(content); Ui::AddSubsectionTitle( content, tr::lng_premium_gifts_summary_subtitle()); } const auto content = box->addRow( object_ptr(box), {}); auto buttonCallback = [=](PremiumFeature section) { stars->setPaused(true); const auto previewBoxShown = [=]( not_null previewBox) { previewBox->boxClosing( ) | rpl::start_with_next(crl::guard(box, [=] { stars->setPaused(false); }), previewBox->lifetime()); }; ShowPremiumPreviewBox( controller->uiShow(), section, previewBoxShown, true); }; Settings::AddSummaryPremium( content, controller, ref, std::move(buttonCallback)); } // Footer. { box->addRow( object_ptr( box, object_ptr( box, tr::lng_premium_gifts_terms( lt_link, tr::lng_payments_terms_link( ) | rpl::map([](const QString &t) { using namespace Ui::Text; return Link(t, u"https://telegram.org/tos"_q); }), lt_policy, tr::lng_premium_gifts_terms_policy( ) | rpl::map([](const QString &t) { using namespace Ui::Text; return Link(t, u"https://telegram.org/privacy"_q); }), Ui::Text::RichLangValue), st::premiumGiftTerms), st::defaultBoxDividerLabelPadding), {}); } // Button. const auto &stButton = st::premiumGiftBox; box->setStyle(stButton); auto raw = Settings::CreateSubscribeButton({ controller, box, [=] { return ref; }, rpl::combine( state->buttonText.events(), state->confirmButtonBusy.value(), state->isPaymentComplete.value() ) | rpl::map([](const QString &text, bool busy, bool paid) { return busy ? QString() : paid ? tr::lng_close(tr::now) : text; }), Ui::Premium::GiftGradientStops(), }); raw->setClickedCallback([=] { if (state->confirmButtonBusy.current()) { return; } if (state->isPaymentComplete.current()) { return box->closeBox(); } auto invoice = api->invoice( users.size(), api->monthsFromPreset(group->current())); invoice.purpose = Payments::InvoicePremiumGiftCodeUsers{ users }; state->confirmButtonBusy = true; const auto show = box->uiShow(); const auto weak = Ui::MakeWeak(box.get()); const auto done = [=](Payments::CheckoutResult result) { if (const auto strong = weak.data()) { strong->window()->setFocus(); state->confirmButtonBusy = false; if (result == Payments::CheckoutResult::Paid) { state->isPaymentComplete = true; Ui::StartFireworks(box->parentWidget()); } } }; Payments::CheckoutProcess::Start(std::move(invoice), done); }); { using namespace Info::Statistics; const auto loadingAnimation = InfiniteRadialAnimationWidget( raw, raw->height() / 2); AddChildToWidgetCenter(raw, loadingAnimation); loadingAnimation->showOn(state->confirmButtonBusy.value()); } auto button = object_ptr::fromRaw(raw); button->resizeToWidth(boxWidth - rect::m::sum::h(stButton.buttonPadding)); box->setShowFinishedCallback([raw = button.data()] { raw->startGlareAnimation(); }); box->addButton(std::move(button)); groupValueChangedCallback(0); } [[nodiscard]] Data::GiftCodeLink MakeGiftCodeLink( not_null session, const QString &slug) { const auto path = u"giftcode/"_q + slug; return { session->createInternalLink(path), session->createInternalLinkFull(path), }; } [[nodiscard]] object_ptr MakeLinkCopyIcon( not_null parent) { auto result = object_ptr(parent); const auto raw = result.data(); raw->paintRequest() | rpl::start_with_next([=] { auto p = QPainter(raw); const auto &icon = st::giveawayGiftCodeLinkCopy; const auto left = (raw->width() - icon.width()) / 2; const auto top = (raw->height() - icon.height()) / 2; icon.paint(p, left, top, raw->width()); }, raw->lifetime()); raw->resize( st::giveawayGiftCodeLinkCopyWidth, st::giveawayGiftCodeLinkHeight); raw->setAttribute(Qt::WA_TransparentForMouseEvents); return result; } [[nodiscard]] tr::phrase GiftDurationPhrase(int months) { return (months < 12) ? tr::lng_premium_gift_duration_months : tr::lng_premium_gift_duration_years; } [[nodiscard]] object_ptr MakePeerTableValue( not_null parent, not_null controller, PeerId id) { auto result = object_ptr(parent); const auto raw = result.data(); const auto &st = st::giveawayGiftCodeUserpic; raw->resize(raw->width(), st.photoSize); const auto peer = controller->session().data().peer(id); const auto userpic = Ui::CreateChild(raw, peer, st); const auto label = Ui::CreateChild( raw, peer->name(), st::giveawayGiftCodeValue); raw->widthValue( ) | rpl::start_with_next([=](int width) { const auto position = st::giveawayGiftCodeNamePosition; label->resizeToNaturalWidth(width - position.x()); label->moveToLeft(position.x(), position.y(), width); const auto top = (raw->height() - userpic->height()) / 2; userpic->moveToLeft(0, top, width); }, label->lifetime()); userpic->setAttribute(Qt::WA_TransparentForMouseEvents); label->setAttribute(Qt::WA_TransparentForMouseEvents); label->setTextColorOverride(st::windowActiveTextFg->c); raw->setClickedCallback([=] { controller->uiShow()->showBox(PrepareShortInfoBox(peer, controller)); }); return result; } void AddTableRow( not_null table, rpl::producer label, object_ptr value, style::margins valueMargins) { table->addRow( object_ptr( table, std::move(label), st::giveawayGiftCodeLabel), std::move(value), st::giveawayGiftCodeLabelMargin, valueMargins); } not_null AddTableRow( not_null table, rpl::producer label, rpl::producer value) { auto widget = object_ptr( table, std::move(value), st::giveawayGiftCodeValue); const auto result = widget.data(); AddTableRow( table, std::move(label), std::move(widget), st::giveawayGiftCodeValueMargin); return result; } void AddTableRow( not_null table, rpl::producer label, not_null controller, PeerId id) { if (!id) { return; } AddTableRow( table, std::move(label), MakePeerTableValue(table, controller, id), st::giveawayGiftCodePeerMargin); } void AddTable( not_null container, not_null controller, const Api::GiftCode ¤t, bool skipReason) { auto table = container->add( object_ptr( container, st::giveawayGiftCodeTable), st::giveawayGiftCodeTableMargin); if (current.from) { AddTableRow( table, tr::lng_gift_link_label_from(), controller, current.from); } if (current.from && current.to) { AddTableRow( table, tr::lng_gift_link_label_to(), controller, current.to); } else if (current.from) { AddTableRow( table, tr::lng_gift_link_label_to(), tr::lng_gift_link_label_to_unclaimed(Ui::Text::WithEntities)); } AddTableRow( table, tr::lng_gift_link_label_gift(), tr::lng_gift_link_gift_premium( lt_duration, GiftDurationValue(current.months) | Ui::Text::ToWithEntities(), Ui::Text::WithEntities)); if (!skipReason && current.from) { const auto reason = AddTableRow( table, tr::lng_gift_link_label_reason(), (current.giveawayId ? ((current.to ? tr::lng_gift_link_reason_giveaway : tr::lng_gift_link_reason_unclaimed)( ) | Ui::Text::ToLink()) : current.giveaway ? ((current.to ? tr::lng_gift_link_reason_giveaway : tr::lng_gift_link_reason_unclaimed)( Ui::Text::WithEntities ) | rpl::type_erased()) : tr::lng_gift_link_reason_chosen(Ui::Text::WithEntities))); reason->setClickHandlerFilter([=](const auto &...) { controller->showPeerHistory( current.from, Window::SectionShow::Way::Forward, current.giveawayId); return false; }); } if (current.date) { AddTableRow( table, tr::lng_gift_link_label_date(), rpl::single(Ui::Text::WithEntities( langDateTime(base::unixtime::parse(current.date))))); } } } // namespace GiftPremiumValidator::GiftPremiumValidator( not_null controller) : _controller(controller) , _api(&_controller->session().mtp()) { } void GiftPremiumValidator::cancel() { _requestId = 0; } void GiftPremiumValidator::showChoosePeerBox(const QString &ref) { if (_manyGiftsLifetime) { return; } using namespace Api; const auto api = _manyGiftsLifetime.make_state( _controller->session().user()); const auto show = _controller->uiShow(); api->request( ) | rpl::start_with_error_done([=](const QString &error) { show->showToast(error); }, [=] { const auto maxAmount = *ranges::max_element(api->availablePresets()); class Controller final : public ContactsBoxController { public: Controller( not_null session, Fn checkErrorCallback) : ContactsBoxController(session) , _checkErrorCallback(std::move(checkErrorCallback)) { } protected: std::unique_ptr createRow( not_null user) override { if (user->isSelf() || user->isBot() || user->isServiceUser() || user->isInaccessible()) { return nullptr; } return ContactsBoxController::createRow(user); } void rowClicked(not_null row) override { const auto checked = !row->checked(); if (checked && _checkErrorCallback && _checkErrorCallback( delegate()->peerListSelectedRowsCount())) { return; } delegate()->peerListSetRowChecked(row, checked); } private: const Fn _checkErrorCallback; }; auto initBox = [=](not_null peersBox) { const auto ignoreClose = peersBox->lifetime().make_state(0); auto process = [=] { const auto selected = peersBox->collectSelectedRows(); const auto users = ranges::views::all( selected ) | ranges::views::transform([](not_null p) { return p->asUser(); }) | ranges::views::filter([](UserData *u) -> bool { return u; }) | ranges::to>>(); if (!users.empty()) { const auto giftBox = show->show( Box(GiftsBox, _controller, users, api, ref)); giftBox->boxClosing( ) | rpl::start_with_next([=] { _manyGiftsLifetime.destroy(); }, giftBox->lifetime()); } (*ignoreClose) = true; peersBox->closeBox(); }; peersBox->setTitle(tr::lng_premium_gift_title()); peersBox->addButton( tr::lng_settings_gift_premium_users_confirm(), std::move(process)); peersBox->addButton(tr::lng_cancel(), [=] { peersBox->closeBox(); }); peersBox->boxClosing( ) | rpl::start_with_next([=] { if (!(*ignoreClose)) { _manyGiftsLifetime.destroy(); } }, peersBox->lifetime()); }; auto listController = std::make_unique( &_controller->session(), [=](int count) { if (count <= maxAmount) { return false; } show->showToast(tr::lng_settings_gift_premium_users_error( tr::now, lt_count, maxAmount)); return true; }); show->showBox( Box( std::move(listController), std::move(initBox)), Ui::LayerOption::KeepOther); }, _manyGiftsLifetime); } void GiftPremiumValidator::showChosenPeerBox( not_null user, const QString &ref) { if (_manyGiftsLifetime) { return; } using namespace Api; const auto api = _manyGiftsLifetime.make_state( _controller->session().user()); const auto show = _controller->uiShow(); api->request( ) | rpl::start_with_error_done([=](const QString &error) { show->showToast(error); }, [=] { const auto users = std::vector>{ user }; const auto giftBox = show->show( Box(GiftsBox, _controller, users, api, ref)); giftBox->boxClosing( ) | rpl::start_with_next([=] { _manyGiftsLifetime.destroy(); }, giftBox->lifetime()); }, _manyGiftsLifetime); } void GiftPremiumValidator::showBox(not_null user) { if (_requestId) { return; } _requestId = _api.request(MTPusers_GetFullUser( user->inputUser )).done([=](const MTPusers_UserFull &result) { if (!_requestId) { // Canceled. return; } _requestId = 0; // _controller->api().processFullPeer(peer, result); _controller->session().data().processUsers(result.data().vusers()); _controller->session().data().processChats(result.data().vchats()); const auto &fullUser = result.data().vfull_user().data(); auto options = GiftOptionFromTL(fullUser); if (!options.empty()) { _controller->show( Box(GiftBox, _controller, user, std::move(options))); } }).fail([=] { _requestId = 0; }).send(); } rpl::producer GiftDurationValue(int months) { return GiftDurationPhrase(months)( lt_count, rpl::single(float64((months < 12) ? months : (months / 12)))); } QString GiftDuration(int months) { return GiftDurationPhrase(months)( tr::now, lt_count, (months < 12) ? months : (months / 12)); } void GiftCodeBox( not_null box, not_null controller, const QString &slug) { struct State { rpl::variable data; rpl::variable used; bool sent = false; }; const auto session = &controller->session(); const auto state = box->lifetime().make_state(State{}); state->data = session->api().premium().giftCodeValue(slug); state->used = state->data.value( ) | rpl::map([=](const Api::GiftCode &data) { return data.used != 0; }); box->setWidth(st::boxWideWidth); box->setStyle(st::giveawayGiftCodeBox); box->setNoContentMargin(true); const auto bar = box->setPinnedToTopContent( object_ptr( box, st::giveawayGiftCodeCover, nullptr, rpl::conditional( state->used.value(), tr::lng_gift_link_used_title(), tr::lng_gift_link_title()), rpl::conditional( state->used.value(), tr::lng_gift_link_used_about(Ui::Text::RichLangValue), tr::lng_gift_link_about(Ui::Text::RichLangValue)), true)); const auto max = st::giveawayGiftCodeTopHeight; bar->setMaximumHeight(max); bar->setMinimumHeight(st::infoLayerTopBarHeight); bar->resize(bar->width(), bar->maximumHeight()); const auto link = MakeGiftCodeLink(&controller->session(), slug); box->addRow( Ui::MakeLinkLabel( box, rpl::single(link.text), rpl::single(link.link), box->uiShow(), MakeLinkCopyIcon(box)), st::giveawayGiftCodeLinkMargin); AddTable(box->verticalLayout(), controller, state->data.current(), false); auto shareLink = tr::lng_gift_link_also_send_link( ) | rpl::map([](const QString &text) { return Ui::Text::Link(text); }); auto richDate = [](const Api::GiftCode &data) { return TextWithEntities{ langDateTime(base::unixtime::parse(data.used)), }; }; const auto footer = box->addRow( object_ptr( box, rpl::conditional( state->used.value(), tr::lng_gift_link_used_footer( lt_date, state->data.value() | rpl::map(richDate), Ui::Text::WithEntities), tr::lng_gift_link_also_send( lt_link, std::move(shareLink), Ui::Text::WithEntities)), st::giveawayGiftCodeFooter), st::giveawayGiftCodeFooterMargin); footer->setClickHandlerFilter([=](const auto &...) { const auto chosen = [=](not_null thread) { const auto content = controller->parentController()->content(); return content->shareUrl( thread, MakeGiftCodeLink(&controller->session(), slug).link, QString()); }; Window::ShowChooseRecipientBox(controller, chosen); return false; }); const auto close = Ui::CreateChild( box.get(), st::boxTitleClose); close->setClickedCallback([=] { box->closeBox(); }); box->widthValue( ) | rpl::start_with_next([=](int width) { close->moveToRight(0, 0); }, box->lifetime()); const auto button = box->addButton(rpl::conditional( state->used.value(), tr::lng_box_ok(), tr::lng_gift_link_use() ), [=] { if (state->used.current()) { box->closeBox(); } else if (!state->sent) { state->sent = true; const auto done = crl::guard(box, [=](const QString &error) { if (error.isEmpty()) { auto copy = state->data.current(); copy.used = base::unixtime::now(); state->data = std::move(copy); Ui::StartFireworks(box->parentWidget()); } else { box->uiShow()->showToast(error); } }); controller->session().api().premium().applyGiftCode(slug, done); } }); const auto buttonPadding = st::giveawayGiftCodeBox.buttonPadding; const auto buttonWidth = st::boxWideWidth - buttonPadding.left() - buttonPadding.right(); button->widthValue() | rpl::filter([=] { return (button->widthNoMargins() != buttonWidth); }) | rpl::start_with_next([=] { button->resizeToWidth(buttonWidth); }, button->lifetime()); } void GiftCodePendingBox( not_null box, not_null controller, const Api::GiftCode &data) { box->setWidth(st::boxWideWidth); box->setStyle(st::giveawayGiftCodeBox); box->setNoContentMargin(true); { const auto peerTo = controller->session().data().peer(data.to); const auto clickContext = [=, weak = base::make_weak(controller)] { if (const auto strong = weak.get()) { strong->uiShow()->showBox( PrepareShortInfoBox(peerTo, strong)); } return QVariant(); }; const auto &st = st::giveawayGiftCodeCover; const auto resultToName = st.about.style.font->elided( peerTo->shortName(), st.about.minWidth / 2, Qt::ElideMiddle); const auto bar = box->setPinnedToTopContent( object_ptr( box, st, clickContext, tr::lng_gift_link_title(), tr::lng_gift_link_pending_about( lt_user, rpl::single(Ui::Text::Link(resultToName)), Ui::Text::RichLangValue), true)); const auto max = st::giveawayGiftCodeTopHeight; bar->setMaximumHeight(max); bar->setMinimumHeight(st::infoLayerTopBarHeight); bar->resize(bar->width(), bar->maximumHeight()); } { const auto linkLabel = box->addRow( Ui::MakeLinkLabel(box, nullptr, nullptr, nullptr, nullptr), st::giveawayGiftCodeLinkMargin); const auto spoiler = Ui::CreateChild(linkLabel); spoiler->lifetime().make_state([=] { spoiler->update(); })->start(); linkLabel->sizeValue( ) | rpl::start_with_next([=](const QSize &s) { spoiler->setGeometry(Rect(s)); }, spoiler->lifetime()); const auto spoilerCached = Ui::SpoilerMessCached( Ui::DefaultTextSpoilerMask(), st::giveawayGiftCodeLink.textFg->c); const auto textHeight = st::giveawayGiftCodeLink.style.font->height; spoiler->paintRequest( ) | rpl::start_with_next([=] { auto p = QPainter(spoiler); const auto rect = spoiler->rect(); const auto r = rect - QMargins( st::boxRowPadding.left(), (rect.height() - textHeight) / 2, st::boxRowPadding.right(), (rect.height() - textHeight) / 2); Ui::FillSpoilerRect(p, r, spoilerCached.frame()); }, spoiler->lifetime()); spoiler->setClickedCallback([show = box->uiShow()] { show->showToast(tr::lng_gift_link_pending_toast(tr::now)); }); spoiler->show(); } AddTable(box->verticalLayout(), controller, data, true); box->addRow( object_ptr( box, tr::lng_gift_link_pending_footer(), st::giveawayGiftCodeFooter), st::giveawayGiftCodeFooterMargin); const auto close = Ui::CreateChild( box.get(), st::boxTitleClose); const auto closeCallback = [=] { box->closeBox(); }; close->setClickedCallback(closeCallback); box->widthValue( ) | rpl::start_with_next([=](int width) { close->moveToRight(0, 0); }, box->lifetime()); const auto button = box->addButton(tr::lng_close(), closeCallback); const auto buttonPadding = st::giveawayGiftCodeBox.buttonPadding; const auto buttonWidth = st::boxWideWidth - buttonPadding.left() - buttonPadding.right(); button->widthValue() | rpl::filter([=] { return (button->widthNoMargins() != buttonWidth); }) | rpl::start_with_next([=] { button->resizeToWidth(buttonWidth); }, button->lifetime()); } void ResolveGiftCode( not_null controller, const QString &slug, PeerId fromId, PeerId toId) { const auto done = [=](Api::GiftCode code) { const auto session = &controller->session(); const auto selfId = session->userPeerId(); if (!code) { controller->showToast(tr::lng_gift_link_expired(tr::now)); } else if (!code.from && fromId == selfId) { code.from = fromId; code.to = toId; const auto self = (fromId == selfId); const auto peer = session->data().peer(self ? toId : fromId); const auto months = code.months; const auto parent = controller->parentController(); Settings::ShowGiftPremium(parent, peer, months, self); } else { controller->uiShow()->showBox(Box(GiftCodeBox, controller, slug)); } }; controller->session().api().premium().checkGiftCode( slug, crl::guard(controller, done)); } void GiveawayInfoBox( not_null box, not_null controller, std::optional start, std::optional results, Api::GiveawayInfo info) { Expects(start || results); using State = Api::GiveawayState; const auto finished = (info.state == State::Finished) || (info.state == State::Refunded); box->setTitle((finished ? tr::lng_prizes_end_title : tr::lng_prizes_how_title)()); const auto first = results ? results->channel->name() : !start->channels.empty() ? start->channels.front()->name() : u"channel"_q; auto text = TextWithEntities(); if (!info.giftCode.isEmpty()) { text.append("\n\n"); text.append(Ui::Text::Bold(tr::lng_prizes_you_won( tr::now, lt_cup, QString::fromUtf8("\xf0\x9f\x8f\x86")))); text.append("\n\n"); } else if (info.state == State::Finished) { text.append("\n\n"); text.append(Ui::Text::Bold(tr::lng_prizes_you_didnt(tr::now))); text.append("\n\n"); } const auto quantity = start ? start->quantity : (results->winnersCount + results->unclaimedCount); const auto months = start ? start->months : results->months; const auto group = results ? results->channel->isMegagroup() : (!start->channels.empty() && start->channels.front()->isMegagroup()); text.append((finished ? tr::lng_prizes_end_text : tr::lng_prizes_how_text)( tr::now, lt_admins, (group ? tr::lng_prizes_admins_group : tr::lng_prizes_admins)( tr::now, lt_count, quantity, lt_channel, Ui::Text::Bold(first), lt_duration, TextWithEntities{ GiftDuration(months) }, Ui::Text::RichLangValue), Ui::Text::RichLangValue)); const auto many = start ? (start->channels.size() > 1) : (results->additionalPeersCount > 0); const auto count = info.winnersCount ? info.winnersCount : quantity; const auto all = start ? start->all : results->all; auto winners = all ? (many ? (group ? tr::lng_prizes_winners_all_of_many_group : tr::lng_prizes_winners_all_of_many) : (group ? tr::lng_prizes_winners_all_of_one_group : tr::lng_prizes_winners_all_of_one))( tr::now, lt_count, count, lt_channel, Ui::Text::Bold(first), Ui::Text::RichLangValue) : (many ? tr::lng_prizes_winners_new_of_many : tr::lng_prizes_winners_new_of_one)( tr::now, lt_count, count, lt_channel, Ui::Text::Bold(first), lt_start_date, Ui::Text::Bold( langDateTime(base::unixtime::parse(info.startDate))), Ui::Text::RichLangValue); const auto additionalPrize = results ? results->additionalPrize : start->additionalPrize; if (!additionalPrize.isEmpty()) { text.append("\n\n").append((group ? tr::lng_prizes_additional_added_group : tr::lng_prizes_additional_added)( tr::now, lt_count, count, lt_channel, Ui::Text::Bold(first), lt_prize, TextWithEntities{ additionalPrize }, Ui::Text::RichLangValue)); } const auto untilDate = start ? start->untilDate : results->untilDate; text.append("\n\n").append((finished ? tr::lng_prizes_end_when_finish : tr::lng_prizes_how_when_finish)( tr::now, lt_date, Ui::Text::Bold(langDayOfMonthFull( base::unixtime::parse(untilDate).date())), lt_winners, winners, Ui::Text::RichLangValue)); if (info.activatedCount > 0) { text.append(' ').append(tr::lng_prizes_end_activated( tr::now, lt_count, info.activatedCount, Ui::Text::RichLangValue)); } if (!info.giftCode.isEmpty() || info.state == State::Finished || info.state == State::Preparing) { } else if (info.state != State::Refunded) { if (info.adminChannelId) { const auto channel = controller->session().data().channel( info.adminChannelId); text.append("\n\n").append((channel->isMegagroup() ? tr::lng_prizes_how_no_admin_group : tr::lng_prizes_how_no_admin)( tr::now, lt_channel, Ui::Text::Bold(channel->name()), Ui::Text::RichLangValue)); } else if (info.tooEarlyDate) { const auto channel = controller->session().data().channel( info.adminChannelId); text.append("\n\n").append((channel->isMegagroup() ? tr::lng_prizes_how_no_joined_group : tr::lng_prizes_how_no_joined)( tr::now, lt_date, Ui::Text::Bold( langDateTime( base::unixtime::parse(info.tooEarlyDate))), Ui::Text::RichLangValue)); } else if (!info.disallowedCountry.isEmpty()) { text.append("\n\n").append(tr::lng_prizes_how_no_country( tr::now, Ui::Text::RichLangValue)); } else if (info.participating) { text.append("\n\n").append((many ? tr::lng_prizes_how_yes_joined_many : tr::lng_prizes_how_yes_joined_one)( tr::now, lt_channel, Ui::Text::Bold(first), Ui::Text::RichLangValue)); } else { text.append("\n\n").append((many ? tr::lng_prizes_how_participate_many : tr::lng_prizes_how_participate_one)( tr::now, lt_channel, Ui::Text::Bold(first), lt_date, Ui::Text::Bold(langDayOfMonthFull( base::unixtime::parse(untilDate).date())), Ui::Text::RichLangValue)); } } const auto padding = st::boxPadding; box->addRow( object_ptr( box.get(), rpl::single(std::move(text)), st::boxLabel), { padding.left(), 0, padding.right(), padding.bottom() }); if (info.state == State::Refunded) { const auto wrap = box->addRow( object_ptr>( box.get(), object_ptr( box.get(), (group ? tr::lng_prizes_cancelled_group() : tr::lng_prizes_cancelled()), st::giveawayRefundedLabel), st::giveawayRefundedPadding), { padding.left(), 0, padding.right(), padding.bottom() }); const auto bg = wrap->lifetime().make_state( st::boxRadius, st::attentionBoxButton.textBgOver); wrap->paintRequest() | rpl::start_with_next([=] { auto p = QPainter(wrap); bg->paint(p, wrap->rect()); }, wrap->lifetime()); } if (const auto slug = info.giftCode; !slug.isEmpty()) { box->addButton(tr::lng_prizes_view_prize(), [=] { ResolveGiftCode(controller, slug); }); box->addButton(tr::lng_cancel(), [=] { box->closeBox(); }); } else { box->addButton(tr::lng_close(), [=] { box->closeBox(); }); } } void ResolveGiveawayInfo( not_null controller, not_null peer, MsgId messageId, std::optional start, std::optional results) { const auto show = [=](Api::GiveawayInfo info) { if (!info) { controller->showToast( tr::lng_confirm_phone_link_invalid(tr::now)); } else { controller->uiShow()->showBox( Box(GiveawayInfoBox, controller, start, results, info)); } }; controller->session().api().premium().resolveGiveawayInfo( peer, messageId, crl::guard(controller, show)); }