251 lines
7.1 KiB
C++
251 lines
7.1 KiB
C++
/*
|
|
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/send_credits_box.h"
|
|
|
|
#include "api/api_credits.h"
|
|
#include "apiwrap.h"
|
|
#include "core/ui_integration.h" // Core::MarkedTextContext.
|
|
#include "data/data_credits.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_user.h"
|
|
#include "data/stickers/data_custom_emoji.h"
|
|
#include "history/history.h"
|
|
#include "history/history_item.h"
|
|
#include "info/channel_statistics/boosts/giveaway/boost_badge.h" // InfiniteRadialAnimationWidget.
|
|
#include "lang/lang_keys.h"
|
|
#include "main/main_session.h"
|
|
#include "payments/payments_checkout_process.h"
|
|
#include "payments/payments_form.h"
|
|
#include "settings/settings_credits_graphics.h"
|
|
#include "ui/controls/userpic_button.h"
|
|
#include "ui/effects/premium_graphics.h"
|
|
#include "ui/effects/premium_top_bar.h" // Ui::Premium::ColorizedSvg.
|
|
#include "ui/image/image_prepare.h"
|
|
#include "ui/layers/generic_box.h"
|
|
#include "ui/rect.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/vertical_list.h"
|
|
#include "ui/widgets/buttons.h"
|
|
#include "styles/style_boxes.h"
|
|
#include "styles/style_credits.h"
|
|
#include "styles/style_giveaway.h"
|
|
#include "styles/style_layers.h"
|
|
#include "styles/style_premium.h"
|
|
#include "styles/style_settings.h"
|
|
|
|
namespace Ui {
|
|
|
|
void SendCreditsBox(
|
|
not_null<Ui::GenericBox*> box,
|
|
std::shared_ptr<Payments::CreditsFormData> form,
|
|
Fn<void()> sent) {
|
|
if (!form) {
|
|
return;
|
|
}
|
|
struct State {
|
|
rpl::variable<bool> confirmButtonBusy = false;
|
|
};
|
|
const auto state = box->lifetime().make_state<State>();
|
|
box->setStyle(st::giveawayGiftCodeBox);
|
|
box->setNoContentMargin(true);
|
|
|
|
const auto session = form->invoice.session;
|
|
|
|
const auto photoSize = st::defaultUserpicButton.photoSize;
|
|
|
|
const auto content = box->verticalLayout();
|
|
Ui::AddSkip(content, photoSize / 2);
|
|
|
|
{
|
|
const auto ministarsContainer = Ui::CreateChild<Ui::RpWidget>(box);
|
|
const auto fullHeight = photoSize * 2;
|
|
using MiniStars = Ui::Premium::ColoredMiniStars;
|
|
const auto ministars = box->lifetime().make_state<MiniStars>(
|
|
ministarsContainer,
|
|
false,
|
|
Ui::Premium::MiniStars::Type::BiStars);
|
|
ministars->setColorOverride(Ui::Premium::CreditsIconGradientStops());
|
|
|
|
ministarsContainer->paintRequest(
|
|
) | rpl::start_with_next([=] {
|
|
auto p = QPainter(ministarsContainer);
|
|
ministars->paint(p);
|
|
}, ministarsContainer->lifetime());
|
|
|
|
box->widthValue(
|
|
) | rpl::start_with_next([=](int width) {
|
|
ministarsContainer->resize(width, fullHeight);
|
|
const auto w = fullHeight / 3 * 2;
|
|
ministars->setCenter(QRect(
|
|
(width - w) / 2,
|
|
(fullHeight - w) / 2,
|
|
w,
|
|
w));
|
|
}, ministarsContainer->lifetime());
|
|
}
|
|
|
|
const auto bot = session->data().user(form->botId);
|
|
|
|
if (form->photo) {
|
|
box->addRow(object_ptr<Ui::CenterWrap<>>(
|
|
content,
|
|
Settings::HistoryEntryPhoto(content, form->photo, photoSize)));
|
|
} else {
|
|
const auto widget = box->addRow(
|
|
object_ptr<Ui::CenterWrap<>>(
|
|
content,
|
|
object_ptr<Ui::UserpicButton>(
|
|
content,
|
|
bot,
|
|
st::defaultUserpicButton)));
|
|
widget->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
}
|
|
|
|
Ui::AddSkip(content);
|
|
box->addRow(object_ptr<Ui::CenterWrap<>>(
|
|
box,
|
|
object_ptr<Ui::FlatLabel>(
|
|
box,
|
|
tr::lng_credits_box_out_title(),
|
|
st::settingsPremiumUserTitle)));
|
|
Ui::AddSkip(content);
|
|
box->addRow(object_ptr<Ui::CenterWrap<>>(
|
|
box,
|
|
object_ptr<Ui::FlatLabel>(
|
|
box,
|
|
tr::lng_credits_box_out_sure(
|
|
lt_count,
|
|
rpl::single(form->invoice.amount) | tr::to_count(),
|
|
lt_text,
|
|
rpl::single(TextWithEntities{ form->title }),
|
|
lt_bot,
|
|
rpl::single(TextWithEntities{ bot->name() }),
|
|
Ui::Text::RichLangValue),
|
|
st::creditsBoxAbout)));
|
|
Ui::AddSkip(content);
|
|
Ui::AddSkip(content);
|
|
|
|
const auto button = box->addButton(rpl::single(QString()), [=] {
|
|
if (state->confirmButtonBusy.current()) {
|
|
return;
|
|
}
|
|
state->confirmButtonBusy = true;
|
|
session->api().request(
|
|
MTPpayments_SendStarsForm(
|
|
MTP_flags(0),
|
|
MTP_long(form->formId),
|
|
form->inputInvoice)
|
|
).done([=](auto result) {
|
|
state->confirmButtonBusy = false;
|
|
box->closeBox();
|
|
sent();
|
|
}).fail([=](const MTP::Error &error) {
|
|
state->confirmButtonBusy = false;
|
|
box->uiShow()->showToast(error.type());
|
|
}).send();
|
|
});
|
|
{
|
|
using namespace Info::Statistics;
|
|
const auto loadingAnimation = InfiniteRadialAnimationWidget(
|
|
button,
|
|
st::giveawayGiftCodeStartButton.height / 2);
|
|
AddChildToWidgetCenter(button.data(), loadingAnimation);
|
|
loadingAnimation->showOn(state->confirmButtonBusy.value());
|
|
}
|
|
{
|
|
const auto emojiMargin = QMargins(
|
|
0,
|
|
-st::moderateBoxExpandInnerSkip,
|
|
0,
|
|
0);
|
|
const auto buttonEmoji = Ui::Text::SingleCustomEmoji(
|
|
session->data().customEmojiManager().registerInternalEmoji(
|
|
st::settingsPremiumIconStar,
|
|
emojiMargin,
|
|
true));
|
|
auto buttonText = tr::lng_credits_box_out_confirm(
|
|
lt_count,
|
|
rpl::single(form->invoice.amount) | tr::to_count(),
|
|
lt_emoji,
|
|
rpl::single(buttonEmoji),
|
|
Ui::Text::RichLangValue);
|
|
const auto buttonLabel = Ui::CreateChild<Ui::FlatLabel>(
|
|
button,
|
|
rpl::single(QString()),
|
|
st::defaultFlatLabel);
|
|
std::move(
|
|
buttonText
|
|
) | rpl::start_with_next([=](const TextWithEntities &text) {
|
|
buttonLabel->setMarkedText(
|
|
text,
|
|
Core::MarkedTextContext{
|
|
.session = session,
|
|
.customEmojiRepaint = [=] { buttonLabel->update(); },
|
|
});
|
|
}, buttonLabel->lifetime());
|
|
buttonLabel->setTextColorOverride(
|
|
box->getDelegate()->style().button.textFg->c);
|
|
button->sizeValue(
|
|
) | rpl::start_with_next([=](const QSize &size) {
|
|
buttonLabel->moveToLeft(
|
|
(size.width() - buttonLabel->width()) / 2,
|
|
(size.height() - buttonLabel->height()) / 2);
|
|
}, buttonLabel->lifetime());
|
|
buttonLabel->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
state->confirmButtonBusy.value(
|
|
) | rpl::start_with_next([=](bool busy) {
|
|
buttonLabel->setVisible(!busy);
|
|
}, buttonLabel->lifetime());
|
|
}
|
|
|
|
const auto buttonWidth = st::boxWidth
|
|
- rect::m::sum::h(st::giveawayGiftCodeBox.buttonPadding);
|
|
button->widthValue() | rpl::filter([=] {
|
|
return (button->widthNoMargins() != buttonWidth);
|
|
}) | rpl::start_with_next([=] {
|
|
button->resizeToWidth(buttonWidth);
|
|
}, button->lifetime());
|
|
|
|
{
|
|
const auto close = Ui::CreateChild<Ui::IconButton>(
|
|
box.get(),
|
|
st::boxTitleClose);
|
|
close->setClickedCallback([=] {
|
|
box->closeBox();
|
|
});
|
|
box->widthValue(
|
|
) | rpl::start_with_next([=](int width) {
|
|
close->moveToRight(0, 0);
|
|
close->raise();
|
|
}, close->lifetime());
|
|
}
|
|
|
|
{
|
|
const auto balance = Settings::AddBalanceWidget(
|
|
content,
|
|
session->creditsValue(),
|
|
false);
|
|
const auto api = balance->lifetime().make_state<Api::CreditsStatus>(
|
|
session->user());
|
|
api->request({}, [=](Data::CreditsStatusSlice slice) {
|
|
session->setCredits(slice.balance);
|
|
});
|
|
rpl::combine(
|
|
balance->sizeValue(),
|
|
content->sizeValue()
|
|
) | rpl::start_with_next([=](const QSize &, const QSize &) {
|
|
balance->moveToLeft(
|
|
st::creditsHistoryRightSkip * 2,
|
|
st::creditsHistoryRightSkip);
|
|
balance->update();
|
|
}, balance->lifetime());
|
|
}
|
|
}
|
|
|
|
} // namespace Ui
|