/*
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 "settings/settings_premium.h"
#include "boxes/premium_preview_box.h"
#include "boxes/sticker_set_box.h"
#include "chat_helpers/stickers_lottie.h" // LottiePlayerFromDocument.
#include "core/application.h"
#include "core/click_handler_types.h"
#include "core/local_url_handlers.h" // Core::TryConvertUrlToLocal.
#include "core/ui_integration.h" // MarkedTextContext.
#include "data/data_document.h"
#include "data/data_document_media.h"
#include "data/data_peer_values.h"
#include "data/data_session.h"
#include "data/stickers/data_custom_emoji.h" // SerializeCustomEmojiId.
#include "data/stickers/data_stickers.h"
#include "history/view/media/history_view_sticker.h" // EmojiSize.
#include "history/view/media/history_view_sticker_player.h"
#include "info/info_wrap_widget.h" // Info::Wrap.
#include "info/profile/info_profile_values.h"
#include "info/settings/info_settings_widget.h" // SectionCustomTopBarData.
#include "lang/lang_keys.h"
#include "main/main_account.h"
#include "main/main_app_config.h"
#include "main/main_session.h"
#include "settings/settings_common.h"
#include "settings/settings_premium.h"
#include "ui/abstract_button.h"
#include "ui/basic_click_handlers.h"
#include "ui/color_contrast.h"
#include "ui/effects/gradient.h"
#include "ui/effects/premium_graphics.h"
#include "ui/effects/premium_stars_colored.h"
#include "ui/layers/generic_box.h"
#include "ui/text/format_values.h"
#include "ui/text/text_utilities.h"
#include "ui/text/text_utilities.h"
#include "ui/widgets/checkbox.h" // Ui::RadiobuttonGroup.
#include "ui/widgets/gradient_round_button.h"
#include "ui/widgets/labels.h"
#include "ui/wrap/fade_wrap.h"
#include "ui/wrap/padding_wrap.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "window/window_controller.h"
#include "window/window_session_controller.h"
#include "base/unixtime.h"
#include "apiwrap.h"
#include "api/api_premium.h"
#include "styles/style_boxes.h"
#include "styles/style_premium.h"
#include "styles/style_info.h"
#include "styles/style_intro.h"
#include "styles/style_layers.h"
#include "styles/style_settings.h"
namespace Settings {
namespace {
using SectionCustomTopBarData = Info::Settings::SectionCustomTopBarData;
constexpr auto kBodyAnimationPart = 0.90;
constexpr auto kTitleAdditionalScale = 0.15;
constexpr auto kMinAcceptableContrast = 4.5; // 1.14;
[[nodiscard]] QString Svg() {
return u":/gui/icons/settings/star.svg"_q;
}
[[nodiscard]] QByteArray ColorizedSvg() {
auto f = QFile(Svg());
if (!f.open(QIODevice::ReadOnly)) {
return QByteArray();
}
auto content = qs(f.readAll());
auto stops = [] {
auto s = QString();
for (const auto &stop : Ui::Premium::ButtonGradientStops()) {
s += QString("")
.arg(QString::number(stop.first), stop.second.name());
}
return s;
}();
const auto color = QString("%5")
.arg(0)
.arg(1)
.arg(1)
.arg(0)
.arg(std::move(stops));
content.replace(u"gradientPlaceholder"_q, color);
content.replace(u"#fff"_q, u"url(#Gradient2)"_q);
f.close();
return content.toUtf8();
}
[[nodiscard]] Data::SubscriptionOptions SubscriptionOptionsForRows(
Data::SubscriptionOptions result) {
for (auto &option : result) {
const auto total = option.costTotal;
const auto perMonth = option.costPerMonth;
option.costTotal = tr::lng_premium_gift_per(
tr::now,
lt_cost,
perMonth);
option.costPerMonth = tr::lng_premium_subscribe_total(
tr::now,
lt_cost,
total);
if (option.duration == tr::lng_months(tr::now, lt_count, 1)) {
option.costPerMonth = QString();
option.duration = tr::lng_premium_subscribe_months_1(tr::now);
} else if (option.duration == tr::lng_months(tr::now, lt_count, 6)) {
option.duration = tr::lng_premium_subscribe_months_6(tr::now);
} else if (option.duration == tr::lng_years(tr::now, lt_count, 1)) {
option.duration = tr::lng_premium_subscribe_months_12(tr::now);
}
}
return result;
}
[[nodiscard]] int TopTransitionSkip() {
return (st::settingsButton.padding.top()
+ st::settingsPremiumRowTitlePadding.top()) / 2;
}
namespace Ref {
namespace Gift {
struct Data {
PeerId peerId;
int months;
bool me;
explicit operator bool() const {
return peerId != 0;
}
};
[[nodiscard]] QString Serialize(const Data &gift) {
return QString::number(gift.peerId.value)
+ ':'
+ QString::number(gift.months)
+ ':'
+ QString::number(gift.me ? 1 : 0);
}
[[nodiscard]] Data Parse(QStringView data) {
const auto components = data.split(':');
if (components.size() != 3) {
return {};
}
return {
.peerId = PeerId(components[0].toULongLong()),
.months = components[1].toInt(),
.me = (components[2].toInt() == 1),
};
}
} // namespace Gift
namespace EmojiStatus {
struct Data {
PeerId peerId;
explicit operator bool() const {
return peerId != 0;
}
};
[[nodiscard]] QString Serialize(const Data &gift) {
return QString("profile_:%1").arg(QString::number(gift.peerId.value));
}
[[nodiscard]] Data Parse(QStringView data) {
if (data.startsWith(u"profile_:"_q)) {
const auto components = data.split(':');
if (components.size() != 2) {
return {};
}
return {
.peerId = PeerId(components[1].toULongLong()),
};
}
return {};
}
} // namespace EmojiStatus
} // namespace Ref
struct Entry {
const style::icon *icon;
rpl::producer title;
rpl::producer description;
std::optional section;
};
using Order = std::vector;
[[nodiscard]] Order FallbackOrder() {
return Order{
u"double_limits"_q,
u"more_upload"_q,
u"faster_download"_q,
u"voice_to_text"_q,
u"no_ads"_q,
u"emoji_status"_q,
u"infinite_reactions"_q,
u"premium_stickers"_q,
u"animated_emoji"_q,
u"advanced_chat_management"_q,
u"profile_badge"_q,
u"animated_userpics"_q,
};
}
[[nodiscard]] base::flat_map EntryMap() {
return base::flat_map{
{
u"double_limits"_q,
Entry{
&st::settingsPremiumIconDouble,
tr::lng_premium_summary_subtitle_double_limits(),
tr::lng_premium_summary_about_double_limits(),
},
},
{
u"more_upload"_q,
Entry{
&st::settingsPremiumIconFiles,
tr::lng_premium_summary_subtitle_more_upload(),
tr::lng_premium_summary_about_more_upload(),
PremiumPreview::MoreUpload,
},
},
{
u"faster_download"_q,
Entry{
&st::settingsPremiumIconSpeed,
tr::lng_premium_summary_subtitle_faster_download(),
tr::lng_premium_summary_about_faster_download(),
PremiumPreview::FasterDownload,
},
},
{
u"voice_to_text"_q,
Entry{
&st::settingsPremiumIconVoice,
tr::lng_premium_summary_subtitle_voice_to_text(),
tr::lng_premium_summary_about_voice_to_text(),
PremiumPreview::VoiceToText,
},
},
{
u"no_ads"_q,
Entry{
&st::settingsPremiumIconChannelsOff,
tr::lng_premium_summary_subtitle_no_ads(),
tr::lng_premium_summary_about_no_ads(),
PremiumPreview::NoAds,
},
},
{
u"emoji_status"_q,
Entry{
&st::settingsPremiumIconStatus,
tr::lng_premium_summary_subtitle_emoji_status(),
tr::lng_premium_summary_about_emoji_status(),
PremiumPreview::EmojiStatus,
},
},
{
u"infinite_reactions"_q,
Entry{
&st::settingsPremiumIconLike,
tr::lng_premium_summary_subtitle_infinite_reactions(),
tr::lng_premium_summary_about_infinite_reactions(),
PremiumPreview::InfiniteReactions,
},
},
{
u"premium_stickers"_q,
Entry{
&st::settingsIconStickers,
tr::lng_premium_summary_subtitle_premium_stickers(),
tr::lng_premium_summary_about_premium_stickers(),
PremiumPreview::Stickers,
},
},
{
u"animated_emoji"_q,
Entry{
&st::settingsIconEmoji,
tr::lng_premium_summary_subtitle_animated_emoji(),
tr::lng_premium_summary_about_animated_emoji(),
PremiumPreview::AnimatedEmoji,
},
},
{
u"advanced_chat_management"_q,
Entry{
&st::settingsIconChat,
tr::lng_premium_summary_subtitle_advanced_chat_management(),
tr::lng_premium_summary_about_advanced_chat_management(),
PremiumPreview::AdvancedChatManagement,
},
},
{
u"profile_badge"_q,
Entry{
&st::settingsPremiumIconStar,
tr::lng_premium_summary_subtitle_profile_badge(),
tr::lng_premium_summary_about_profile_badge(),
PremiumPreview::ProfileBadge,
},
},
{
u"animated_userpics"_q,
Entry{
&st::settingsPremiumIconPlay,
tr::lng_premium_summary_subtitle_animated_userpics(),
tr::lng_premium_summary_about_animated_userpics(),
PremiumPreview::AnimatedUserpics,
},
},
};
}
void SendAppLog(
not_null session,
const QString &type,
const MTPJSONValue &data) {
const auto now = double(base::unixtime::now())
+ (QTime::currentTime().msec() / 1000.);
session->api().request(MTPhelp_SaveAppLog(
MTP_vector(1, MTP_inputAppEvent(
MTP_double(now),
MTP_string(type),
MTP_long(0),
data
))
)).send();
}
[[nodiscard]] QString ResolveRef(const QString &ref) {
return ref.isEmpty() ? "settings" : ref;
}
void SendScreenShow(
not_null controller,
const std::vector &order,
const QString &ref) {
auto list = QVector();
list.reserve(order.size());
for (const auto &element : order) {
list.push_back(MTP_jsonString(MTP_string(element)));
}
auto values = QVector{
MTP_jsonObjectValue(
MTP_string("premium_promo_order"),
MTP_jsonArray(MTP_vector(std::move(list)))),
MTP_jsonObjectValue(
MTP_string("source"),
MTP_jsonString(MTP_string(ResolveRef(ref)))),
};
const auto data = MTP_jsonObject(
MTP_vector(std::move(values)));
SendAppLog(
&controller->session(),
"premium.promo_screen_show",
data);
}
void SendScreenAccept(not_null controller) {
SendAppLog(
&controller->session(),
"premium.promo_screen_accept",
MTP_jsonNull());
}
class TopBarAbstract : public Ui::RpWidget {
public:
using Ui::RpWidget::RpWidget;
void setRoundEdges(bool value);
virtual void setPaused(bool paused) = 0;
virtual void setTextPosition(int x, int y) = 0;
protected:
void paintEdges(QPainter &p, const QBrush &brush) const;
void paintEdges(QPainter &p) const;
[[nodiscard]] QRectF starRect(
float64 topProgress,
float64 sizeProgress) const;
[[nodiscard]] bool isDark() const;
void computeIsDark();
private:
bool _roundEdges = true;
bool _isDark = false;
};
void TopBarAbstract::setRoundEdges(bool value) {
_roundEdges = value;
update();
}
void TopBarAbstract::paintEdges(QPainter &p, const QBrush &brush) const {
const auto r = rect();
if (_roundEdges) {
PainterHighQualityEnabler hq(p);
const auto radius = st::boxRadius;
p.setPen(Qt::NoPen);
p.setBrush(brush);
p.drawRoundedRect(
r + QMargins{ 0, 0, 0, radius + 1 },
radius,
radius);
} else {
p.fillRect(r, brush);
}
}
void TopBarAbstract::paintEdges(QPainter &p) const {
paintEdges(p, st::boxBg);
if (isDark()) {
paintEdges(p, st::shadowFg);
paintEdges(p, st::shadowFg);
}
}
QRectF TopBarAbstract::starRect(
float64 topProgress,
float64 sizeProgress) const {
const auto starSize = st::settingsPremiumStarSize * sizeProgress;
return QRectF(
QPointF(
(width() - starSize.width()) / 2,
st::settingsPremiumStarTopSkip * topProgress),
starSize);
};
bool TopBarAbstract::isDark() const {
return _isDark;
}
void TopBarAbstract::computeIsDark() {
const auto contrast = Ui::CountContrast(
st::boxBg->c,
st::premiumButtonFg->c);
_isDark = (contrast > kMinAcceptableContrast);
}
class EmojiStatusTopBar final {
public:
EmojiStatusTopBar(
not_null document,
Fn callback,
QSizeF size);
void setCenter(QPointF position);
void setPaused(bool paused);
void paint(QPainter &p);
private:
[[nodiscard]] QPixmap paintedPixmap(const QSize &size) const;
void resolveIsColored();
QRectF _rect;
std::shared_ptr _media;
std::unique_ptr _player;
bool _paused = false;
bool _isColored = false;
bool _isColoredResolved = false;
rpl::lifetime _lifetime;
};
EmojiStatusTopBar::EmojiStatusTopBar(
not_null document,
Fn callback,
QSizeF size)
: _rect(QPointF(), size) {
const auto sticker = document->sticker();
Assert(sticker != nullptr);
_media = document->createMediaView();
_media->checkStickerLarge();
_media->goodThumbnailWanted();
rpl::single() | rpl::then(
document->owner().session().downloaderTaskFinished()
) | rpl::start_with_next([=] {
if (!_media->loaded()) {
return;
}
_lifetime.destroy();
if (sticker->isLottie()) {
_player = std::make_unique(
ChatHelpers::LottiePlayerFromDocument(
_media.get(),
ChatHelpers::StickerLottieSize::EmojiInteractionReserved7, //
size.toSize(),
Lottie::Quality::High));
} else if (sticker->isWebm()) {
_player = std::make_unique(
_media->owner()->location(),
_media->bytes(),
size.toSize());
}
if (_player) {
_player->setRepaintCallback([=] { callback(_rect.toRect()); });
} else {
callback(_rect.toRect());
}
}, _lifetime);
}
void EmojiStatusTopBar::setCenter(QPointF position) {
const auto size = _rect.size();
const auto shift = QPointF(size.width() / 2., size.height() / 2.);
_rect = QRectF(QPointF(position - shift), QPointF(position + shift));
}
void EmojiStatusTopBar::setPaused(bool paused) {
_paused = paused;
}
QPixmap EmojiStatusTopBar::paintedPixmap(const QSize &size) const {
const auto good = _media->goodThumbnail();
if (const auto image = _media->getStickerLarge()) {
return image->pix(size);
} else if (good) {
return good->pix(size);
} else if (const auto thumbnail = _media->thumbnail()) {
return thumbnail->pix(size, { .options = Images::Option::Blur });
}
return QPixmap();
}
void EmojiStatusTopBar::resolveIsColored() {
if (_isColoredResolved) {
return;
}
const auto document = _media->owner();
const auto manager = &document->owner().customEmojiManager();
const auto coloredSetId = manager->coloredSetId();
if (!coloredSetId) {
return;
}
_isColoredResolved = true;
const auto sticker = document->sticker();
_isColored = sticker && (sticker->set.id == coloredSetId);
}
void EmojiStatusTopBar::paint(QPainter &p) {
if (_player) {
if (_player->ready()) {
resolveIsColored();
const auto frame = _player->frame(
_rect.size().toSize(),
(_isColored
? st::profileVerifiedCheckBg->c
: QColor(0, 0, 0, 0)),
false,
crl::now(),
_paused);
p.drawImage(_rect, frame.image);
if (!_paused) {
_player->markFrameShown();
}
}
} else if (_media) {
p.drawPixmap(_rect.topLeft(), paintedPixmap(_rect.size().toSize()));
}
}
class TopBarUser final : public TopBarAbstract {
public:
TopBarUser(
not_null parent,
not_null controller,
not_null peer,
rpl::producer<> showFinished);
void setPaused(bool paused) override;
void setTextPosition(int x, int y) override;
protected:
void paintEvent(QPaintEvent *e) override;
void resizeEvent(QResizeEvent *e) override;
private:
void updateTitle(
DocumentData *document,
TextWithEntities name,
not_null controller);
void updateAbout(DocumentData *document) const;
object_ptr _content;
object_ptr _title;
object_ptr _about;
Ui::Premium::ColoredMiniStars _ministars;
struct {
object_ptr widget;
Ui::Text::String text;
Ui::Animations::Simple animation;
bool shown = false;
QPoint position;
} _smallTop;
std::unique_ptr _emojiStatus;
QImage _imageStar;
QRectF _starRect;
};
TopBarUser::TopBarUser(
not_null parent,
not_null controller,
not_null peer,
rpl::producer<> showFinished)
: TopBarAbstract(parent)
, _content(this)
, _title(_content, st::settingsPremiumUserTitle)
, _about(_content, st::settingsPremiumUserAbout)
, _ministars(_content)
, _smallTop({
.widget = object_ptr(this),
.text = Ui::Text::String(
st::boxTitle.style,
tr::lng_premium_summary_title(tr::now)),
}) {
_starRect = TopBarAbstract::starRect(1., 1.);
rpl::single() | rpl::then(
style::PaletteChanged()
) | rpl::start_with_next([=] {
TopBarAbstract::computeIsDark();
update();
}, lifetime());
auto documentValue = Info::Profile::EmojiStatusIdValue(
peer
) | rpl::map([=](DocumentId id) -> DocumentData* {
const auto document = id
? controller->session().data().document(id).get()
: nullptr;
return (document && document->sticker()) ? document : nullptr;
});
rpl::combine(
std::move(documentValue),
Info::Profile::NameValue(peer)
) | rpl::start_with_next([=](
DocumentData *document,
TextWithEntities name) {
if (document) {
_emojiStatus = std::make_unique(
document,
[=](QRect r) { update(std::move(r)); },
HistoryView::Sticker::EmojiSize());
_imageStar = QImage();
} else {
auto svg = QSvgRenderer(Svg());
const auto size = _starRect.size().toSize();
auto frame = QImage(
size * style::DevicePixelRatio(),
QImage::Format_ARGB32_Premultiplied);
frame.setDevicePixelRatio(style::DevicePixelRatio());
auto mask = frame;
mask.fill(Qt::transparent);
{
Painter p(&mask);
auto gradient = QLinearGradient(
0,
size.height(),
size.width(),
0);
gradient.setStops(Ui::Premium::ButtonGradientStops());
p.setPen(Qt::NoPen);
p.setBrush(gradient);
p.drawRect(0, 0, size.width(), size.height());
}
frame.fill(Qt::transparent);
{
Painter q(&frame);
svg.render(&q, QRect(QPoint(), size));
q.setCompositionMode(QPainter::CompositionMode_SourceIn);
q.drawImage(0, 0, mask);
}
_imageStar = std::move(frame);
_emojiStatus = nullptr;
}
updateTitle(document, name, controller);
updateAbout(document);
auto event = QResizeEvent(size(), size());
resizeEvent(&event);
update();
}, lifetime());
rpl::combine(
_title->sizeValue(),
_about->sizeValue(),
_content->sizeValue()
) | rpl::start_with_next([=](
const QSize &titleSize,
const QSize &aboutSize,
const QSize &size) {
const auto rect = TopBarAbstract::starRect(1., 1.);
const auto &padding = st::settingsPremiumUserTitlePadding;
_title->moveToLeft(
(size.width() - titleSize.width()) / 2,
rect.top() + rect.height() + padding.top());
_about->moveToLeft(
(size.width() - aboutSize.width()) / 2,
_title->y() + titleSize.height() + padding.bottom());
const auto aboutBottom = _about->y() + _about->height();
const auto height = (aboutBottom > st::settingsPremiumUserHeight)
? aboutBottom + padding.bottom()
: st::settingsPremiumUserHeight;
{
const auto was = maximumHeight();
const auto now = height;
if (was != now) {
setMaximumHeight(now);
if (was == size.height()) {
resize(size.width(), now);
}
}
}
_content->resize(size.width(), maximumHeight());
}, lifetime());
const auto smallTopShadow = Ui::CreateChild(
_smallTop.widget.data());
smallTopShadow->setDuration(st::infoTopBarDuration);
rpl::combine(
rpl::single(
false
) | rpl::then(std::move(showFinished) | rpl::map_to(true)),
sizeValue()
) | rpl::start_with_next([=](bool showFinished, const QSize &size) {
_content->resize(size.width(), maximumHeight());
const auto skip = TopTransitionSkip();
_content->moveToLeft(0, size.height() - _content->height() - skip);
_smallTop.widget->resize(size.width(), minimumHeight());
smallTopShadow->resizeToWidth(size.width());
smallTopShadow->moveToLeft(
0,
_smallTop.widget->height() - smallTopShadow->height());
const auto shown = (minimumHeight() * 2 > size.height());
if (_smallTop.shown != shown) {
_smallTop.shown = shown;
if (!showFinished) {
_smallTop.widget->update();
smallTopShadow->toggle(_smallTop.shown, anim::type::instant);
} else {
_smallTop.animation.start(
[=] { _smallTop.widget->update(); },
_smallTop.shown ? 0. : 1.,
_smallTop.shown ? 1. : 0.,
st::infoTopBarDuration);
smallTopShadow->toggle(_smallTop.shown, anim::type::normal);
}
}
}, lifetime());
_smallTop.widget->paintRequest(
) | rpl::start_with_next([=] {
Painter p(_smallTop.widget);
p.setOpacity(_smallTop.animation.value(_smallTop.shown ? 1. : 0.));
TopBarAbstract::paintEdges(p);
p.setPen(st::boxTitleFg);
_smallTop.text.drawLeft(
p,
_smallTop.position.x(),
_smallTop.position.y(),
width(),
width());
}, lifetime());
_content->paintRequest(
) | rpl::start_with_next([=] {
Painter p(_content);
_ministars.paint(p);
if (_emojiStatus) {
_emojiStatus->paint(p);
} else if (!_imageStar.isNull()) {
p.drawImage(_starRect.topLeft(), _imageStar);
}
}, lifetime());
}
void TopBarUser::updateTitle(
DocumentData *document,
TextWithEntities name,
not_null controller) {
if (!document) {
return _title->setMarkedText(
tr::lng_premium_summary_user_title(
tr::now,
lt_user,
std::move(name),
Ui::Text::WithEntities));
}
const auto stickerInfo = document->sticker();
if (!stickerInfo) {
return;
}
const auto owner = &document->owner();
const auto &sets = owner->stickers().sets();
const auto setId = stickerInfo->set.id;
const auto it = sets.find(setId);
if (it == sets.cend()) {
return;
}
const auto set = it->second.get();
const auto coloredId = owner->customEmojiManager().coloredSetId();
const auto text = (set->thumbnailDocumentId ? QChar('0') : QChar())
+ set->title;
const auto linkIndex = 1;
const auto entityEmojiData = Data::SerializeCustomEmojiId(
{ set->thumbnailDocumentId });
const auto entities = EntitiesInText{
{ EntityType::CustomEmoji, 0, 1, entityEmojiData },
Ui::Text::Link(text, linkIndex).entities.front(),
};
auto title = (setId == coloredId)
? tr::lng_premium_emoji_status_title_colored(
tr::now,
lt_user,
std::move(name),
Ui::Text::WithEntities)
: tr::lng_premium_emoji_status_title(
tr::now,
lt_user,
std::move(name),
lt_link,
{ .text = text, .entities = entities, },
Ui::Text::WithEntities);
const auto context = Core::MarkedTextContext{
.session = &controller->session(),
.customEmojiRepaint = [=] { _title->update(); },
};
_title->setMarkedText(std::move(title), context);
auto link = std::make_shared([=,
stickerSetIdentifier = stickerInfo->set] {
setPaused(true);
const auto box = controller->show(
Box(
controller,
stickerSetIdentifier,
Data::StickersType::Emoji),
Ui::LayerOption::KeepOther);
box->boxClosing(
) | rpl::start_with_next(crl::guard(this, [=] {
setPaused(false);
}), box->lifetime());
});
_title->setLink(linkIndex, std::move(link));
}
void TopBarUser::updateAbout(DocumentData *document) const {
_about->setMarkedText((document
? tr::lng_premium_emoji_status_about
: tr::lng_premium_summary_user_about)(
tr::now,
Ui::Text::RichLangValue));
}
void TopBarUser::setPaused(bool paused) {
_ministars.setPaused(paused);
if (_emojiStatus) {
_emojiStatus->setPaused(paused);
}
}
void TopBarUser::setTextPosition(int x, int y) {
_smallTop.position = { x, y };
}
void TopBarUser::paintEvent(QPaintEvent *e) {
Painter p(this);
TopBarAbstract::paintEdges(p);
}
void TopBarUser::resizeEvent(QResizeEvent *e) {
_starRect = TopBarAbstract::starRect(1., 1.);
_ministars.setCenter(_starRect.toRect());
if (_emojiStatus) {
_emojiStatus->setCenter(_starRect.center());
}
}
class TopBar final : public TopBarAbstract {
public:
TopBar(
not_null parent,
not_null controller,
rpl::producer title,
rpl::producer about);
void setPaused(bool paused) override;
void setTextPosition(int x, int y) override;
protected:
void paintEvent(QPaintEvent *e) override;
void resizeEvent(QResizeEvent *e) override;
private:
const style::font &_titleFont;
const style::margins &_titlePadding;
object_ptr _about;
Ui::Premium::ColoredMiniStars _ministars;
QSvgRenderer _star;
struct {
float64 top = 0.;
float64 body = 0.;
float64 title = 0.;
float64 scaleTitle = 0.;
} _progress;
QRectF _starRect;
QPoint _titlePosition;
QPainterPath _titlePath;
};
TopBar::TopBar(
not_null parent,
not_null controller,
rpl::producer title,
rpl::producer about)
: TopBarAbstract(parent)
, _titleFont(st::boxTitle.style.font)
, _titlePadding(st::settingsPremiumTitlePadding)
, _about(this, std::move(about), st::settingsPremiumAbout)
, _ministars(this) {
std::move(
title
) | rpl::start_with_next([=](QString text) {
_titlePath = QPainterPath();
_titlePath.addText(0, _titleFont->ascent, _titleFont, text);
update();
}, lifetime());
_about->setClickHandlerFilter([=](
const ClickHandlerPtr &handler,
Qt::MouseButton button) {
ActivateClickHandler(_about, handler, {
button,
QVariant::fromValue(ClickHandlerContext{
.sessionWindow = base::make_weak(controller.get()),
.botStartAutoSubmit = true,
})
});
return false;
});
rpl::single() | rpl::then(
style::PaletteChanged()
) | rpl::start_with_next([=] {
TopBarAbstract::computeIsDark();
if (!TopBarAbstract::isDark()) {
_star.load(Svg());
_ministars.setColorOverride(st::premiumButtonFg->c);
} else {
_star.load(ColorizedSvg());
_ministars.setColorOverride(std::nullopt);
}
auto event = QResizeEvent(size(), size());
resizeEvent(&event);
}, lifetime());
}
void TopBar::setPaused(bool paused) {
_ministars.setPaused(paused);
}
void TopBar::setTextPosition(int x, int y) {
_titlePosition = { x, y };
}
void TopBar::resizeEvent(QResizeEvent *e) {
const auto progress = (e->size().height() - minimumHeight())
/ float64(maximumHeight() - minimumHeight());
_progress.top = 1. -
std::clamp(
(1. - progress) / kBodyAnimationPart,
0.,
1.);
_progress.body = _progress.top;
_progress.title = 1. - progress;
_progress.scaleTitle = 1. + kTitleAdditionalScale * progress;
_ministars.setCenter(starRect(_progress.top, 1.).toRect());
_starRect = starRect(_progress.top, _progress.body);
const auto &padding = st::boxRowPadding;
const auto availableWidth = width() - padding.left() - padding.right();
const auto titleTop = _starRect.top()
+ _starRect.height()
+ _titlePadding.top();
const auto titlePathRect = _titlePath.boundingRect();
const auto aboutTop = titleTop
+ titlePathRect.height()
+ _titlePadding.bottom();
_about->resizeToWidth(availableWidth);
_about->moveToLeft(padding.left(), aboutTop);
_about->setOpacity(_progress.body);
Ui::RpWidget::resizeEvent(e);
}
void TopBar::paintEvent(QPaintEvent *e) {
Painter p(this);
p.fillRect(e->rect(), Qt::transparent);
const auto r = rect();
if (!TopBarAbstract::isDark()) {
const auto gradientPointTop = r.height() / 3. * 2.;
auto gradient = QLinearGradient(
QPointF(0, gradientPointTop),
QPointF(r.width(), r.height() - gradientPointTop));
gradient.setStops(Ui::Premium::ButtonGradientStops());
TopBarAbstract::paintEdges(p, gradient);
} else {
TopBarAbstract::paintEdges(p);
}
p.setOpacity(_progress.body);
p.translate(_starRect.center());
p.scale(_progress.body, _progress.body);
p.translate(-_starRect.center());
if (_progress.top) {
_ministars.paint(p);
}
p.resetTransform();
_star.render(&p, _starRect);
p.setPen(st::premiumButtonFg);
const auto titlePathRect = _titlePath.boundingRect();
// Title.
PainterHighQualityEnabler hq(p);
p.setOpacity(1.);
p.setFont(_titleFont);
const auto fullStarRect = starRect(1., 1.);
const auto fullTitleTop = fullStarRect.top()
+ fullStarRect.height()
+ _titlePadding.top();
p.translate(
anim::interpolate(
(width() - titlePathRect.width()) / 2,
_titlePosition.x(),
_progress.title),
anim::interpolate(fullTitleTop, _titlePosition.y(), _progress.title));
p.translate(titlePathRect.center());
p.scale(_progress.scaleTitle, _progress.scaleTitle);
p.translate(-titlePathRect.center());
p.fillPath(_titlePath, st::premiumButtonFg);
}
class Premium : public Section {
public:
Premium(
QWidget *parent,
not_null controller);
[[nodiscard]] rpl::producer title() override;
[[nodiscard]] QPointer createPinnedToTop(
not_null parent) override;
[[nodiscard]] QPointer createPinnedToBottom(
not_null parent) override;
void showFinished() override;
[[nodiscard]] bool hasFlexibleTopBar() const override;
void setStepDataReference(std::any &data) override;
[[nodiscard]] rpl::producer<> sectionShowBack() override final;
private:
void setupContent();
void setupSubscriptionOptions(not_null container);
const not_null _controller;
const QString _ref;
QPointer _subscribe;
base::unique_qptr> _back;
base::unique_qptr _close;
rpl::variable _backToggles;
rpl::variable _wrap;
Fn _setPaused;
std::shared_ptr _radioGroup;
rpl::event_stream<> _showBack;
rpl::event_stream<> _showFinished;
rpl::variable _buttonText;
};
Premium::Premium(
QWidget *parent,
not_null controller)
: Section(parent)
, _controller(controller)
, _ref(ResolveRef(controller->premiumRef()))
, _radioGroup(std::make_shared()) {
setupContent();
_controller->session().api().premium().reload();
}
rpl::producer Premium::title() {
return tr::lng_premium_summary_title();
}
bool Premium::hasFlexibleTopBar() const {
return true;
}
rpl::producer<> Premium::sectionShowBack() {
return _showBack.events();
}
void Premium::setStepDataReference(std::any &data) {
const auto my = std::any_cast(&data);
if (my) {
_backToggles = std::move(
my->backButtonEnables
) | rpl::map_to(true);
_wrap = std::move(my->wrapValue);
}
}
void Premium::setupSubscriptionOptions(
not_null container) {
const auto isEmojiStatus = (!!Ref::EmojiStatus::Parse(_ref));
const auto options = container->add(
object_ptr>(
container,
object_ptr(container)));
const auto skip = container->add(
object_ptr>(
container,
object_ptr(container)));
const auto content = options->entity();
AddSkip(content, st::settingsPremiumOptionsPadding.top());
const auto apiPremium = &_controller->session().api().premium();
Ui::Premium::AddGiftOptions(
content,
_radioGroup,
SubscriptionOptionsForRows(apiPremium->subscriptionOptions()),
st::premiumSubscriptionOption,
true);
AddSkip(content, st::settingsPremiumOptionsPadding.bottom());
AddDivider(content);
const auto lastSkip = TopTransitionSkip() * (isEmojiStatus ? 1 : 2);
AddSkip(content, lastSkip - st::settingsSectionSkip);
AddSkip(skip->entity(), lastSkip);
auto toggleOn = rpl::combine(
Data::AmPremiumValue(&_controller->session()),
rpl::single(isEmojiStatus),
apiPremium->statusTextValue(
) | rpl::map([=] {
return apiPremium->subscriptionOptions().size() < 2;
})
) | rpl::map([=](bool premium, bool isEmojiStatus, bool noOptions) {
return !premium && !isEmojiStatus && !noOptions;
});
options->toggleOn(rpl::duplicate(toggleOn), anim::type::instant);
skip->toggleOn(std::move(
toggleOn
) | rpl::map([](bool value) { return !value; }), anim::type::instant);
}
void Premium::setupContent() {
const auto content = Ui::CreateChild(this);
const auto &stDefault = st::settingsButton;
const auto &stLabel = st::defaultFlatLabel;
const auto iconSize = st::settingsPremiumIconDouble.size();
const auto &titlePadding = st::settingsPremiumRowTitlePadding;
const auto &descriptionPadding = st::settingsPremiumRowAboutPadding;
setupSubscriptionOptions(content);
auto entryMap = EntryMap();
auto iconContainers = std::vector();
iconContainers.reserve(int(entryMap.size()));
const auto addRow = [&](Entry &entry) {
const auto labelAscent = stLabel.style.font->ascent;
const auto button = Ui::CreateChild(
content,
rpl::single(QString()));
const auto label = content->add(
object_ptr(
content,
std::move(entry.title) | rpl::map(Ui::Text::Bold),
stLabel),
titlePadding);
label->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto description = content->add(
object_ptr(
content,
std::move(entry.description),
st::boxDividerLabel),
descriptionPadding);
description->setAttribute(Qt::WA_TransparentForMouseEvents);
const auto dummy = Ui::CreateChild(content);
dummy->setAttribute(Qt::WA_TransparentForMouseEvents);
content->sizeValue(
) | rpl::start_with_next([=](const QSize &s) {
dummy->resize(s.width(), iconSize.height());
}, dummy->lifetime());
label->geometryValue(
) | rpl::start_with_next([=](const QRect &r) {
dummy->moveToLeft(0, r.y() + (r.height() - labelAscent));
}, dummy->lifetime());
rpl::combine(
content->widthValue(),
label->heightValue(),
description->heightValue()
) | rpl::start_with_next([=,
topPadding = titlePadding,
bottomPadding = descriptionPadding](
int width,
int topHeight,
int bottomHeight) {
button->resize(
width,
topPadding.top()
+ topHeight
+ topPadding.bottom()
+ bottomPadding.top()
+ bottomHeight
+ bottomPadding.bottom());
}, button->lifetime());
label->topValue(
) | rpl::start_with_next([=, padding = titlePadding.top()](int top) {
button->moveToLeft(0, top - padding);
}, button->lifetime());
const auto arrow = Ui::CreateChild(
button,
st::backButton);
arrow->setIconOverride(
&st::settingsPremiumArrow,
&st::settingsPremiumArrowOver);
arrow->setAttribute(Qt::WA_TransparentForMouseEvents);
button->sizeValue(
) | rpl::start_with_next([=](const QSize &s) {
const auto &point = st::settingsPremiumArrowShift;
arrow->moveToRight(
-point.x(),
point.y() + (s.height() - arrow->height()) / 2);
}, arrow->lifetime());
const auto section = entry.section;
button->setClickedCallback([=, controller = _controller] {
_setPaused(true);
const auto hidden = crl::guard(this, [=] {
_setPaused(false);
});
if (section) {
ShowPremiumPreviewToBuy(controller, *section, hidden);
return;
}
controller->show(Box([=](not_null box) {
DoubledLimitsPreviewBox(box, &controller->session());
box->addTopButton(st::boxTitleClose, [=] {
box->closeBox();
});
Data::AmPremiumValue(
&controller->session()
) | rpl::skip(1) | rpl::start_with_next([=] {
box->closeBox();
}, box->lifetime());
if (controller->session().premium()) {
box->addButton(tr::lng_close(), [=] {
box->closeBox();
});
} else {
const auto button = CreateSubscribeButton({
controller,
box,
[] { return u"double_limits"_q; }
});
box->boxClosing(
) | rpl::start_with_next(hidden, box->lifetime());
box->setShowFinishedCallback([=] {
button->startGlareAnimation();
});
box->setStyle(st::premiumPreviewDoubledLimitsBox);
box->widthValue(
) | rpl::start_with_next([=](int width) {
const auto &padding =
st::premiumPreviewDoubledLimitsBox.buttonPadding;
button->resizeToWidth(width
- padding.left()
- padding.right());
button->moveToLeft(padding.left(), padding.top());
}, button->lifetime());
box->addButton(
object_ptr::fromRaw(button));
}
}));
});
iconContainers.push_back(dummy);
};
auto icons = std::vector();
icons.reserve(int(entryMap.size()));
{
const auto &account = _controller->session().account();
const auto mtpOrder = account.appConfig().get(
"premium_promo_order",
FallbackOrder());
const auto processEntry = [&](Entry &entry) {
icons.push_back(entry.icon);
addRow(entry);
};
for (const auto &key : mtpOrder) {
auto it = entryMap.find(key);
if (it == end(entryMap)) {
continue;
}
processEntry(it->second);
}
SendScreenShow(_controller, mtpOrder, _ref);
}
content->resizeToWidth(content->height());
// Icons.
Assert(iconContainers.size() > 2);
const auto from = iconContainers.front()->y();
const auto to = iconContainers.back()->y() + iconSize.height();
auto gradient = QLinearGradient(0, 0, 0, to - from);
gradient.setStops(Ui::Premium::FullHeightGradientStops());
for (auto i = 0; i < int(icons.size()); i++) {
const auto &iconContainer = iconContainers[i];
const auto pointTop = iconContainer->y() - from;
const auto pointBottom = pointTop + iconContainer->height();
const auto ratioTop = pointTop / float64(to - from);
const auto ratioBottom = pointBottom / float64(to - from);
auto resultGradient = QLinearGradient(
QPointF(),
QPointF(0, pointBottom - pointTop));
resultGradient.setColorAt(
.0,
anim::gradient_color_at(gradient, ratioTop));
resultGradient.setColorAt(
.1,
anim::gradient_color_at(gradient, ratioBottom));
const auto brush = QBrush(resultGradient);
AddButtonIcon(
iconContainer,
stDefault,
{ .icon = icons[i], .backgroundBrush = brush });
}
AddSkip(content, descriptionPadding.bottom());
#if 0
AddSkip(content);
AddDivider(content);
AddSkip(content);
content->add(
object_ptr(
content,
tr::lng_premium_summary_bottom_subtitle(
) | rpl::map(Ui::Text::Bold),
stLabel),
st::settingsSubsectionTitlePadding);
content->add(
object_ptr(
content,
tr::lng_premium_summary_bottom_about(Ui::Text::RichLangValue),
st::aboutLabel),
st::boxRowPadding);
AddSkip(content, stDefault.padding.top() + stDefault.padding.bottom());
#endif
Ui::ResizeFitChild(this, content);
}
QPointer Premium::createPinnedToTop(
not_null parent) {
auto title = _controller->session().premium()
? tr::lng_premium_summary_title()
: rpl::conditional(
Data::AmPremiumValue(&_controller->session()),
tr::lng_premium_summary_title_subscribed(),
tr::lng_premium_summary_title());
auto about = [&]() -> rpl::producer {
const auto gift = Ref::Gift::Parse(_ref);
if (gift) {
auto &data = _controller->session().data();
if (const auto peer = data.peer(gift.peerId)) {
return (gift.me
? tr::lng_premium_summary_subtitle_gift_me
: tr::lng_premium_summary_subtitle_gift)(
lt_count,
rpl::single(float64(gift.months)),
lt_user,
rpl::single(Ui::Text::Bold(peer->name())),
Ui::Text::RichLangValue);
}
}
return rpl::conditional(
Data::AmPremiumValue(&_controller->session()),
_controller->session().api().premium().statusTextValue(),
tr::lng_premium_summary_top_about(Ui::Text::RichLangValue));
}();
const auto emojiStatusData = Ref::EmojiStatus::Parse(_ref);
const auto isEmojiStatus = (!!emojiStatusData);
auto peerWithPremium = [&]() -> PeerData* {
if (isEmojiStatus) {
auto &data = _controller->session().data();
if (const auto peer = data.peer(emojiStatusData.peerId)) {
return peer;
}
}
return nullptr;
}();
const auto content = [&]() -> TopBarAbstract* {
if (peerWithPremium) {
return Ui::CreateChild(
parent.get(),
_controller,
peerWithPremium,
_showFinished.events());
}
return Ui::CreateChild(
parent.get(),
_controller,
std::move(title),
std::move(about));
}();
_setPaused = [=](bool paused) {
content->setPaused(paused);
if (_subscribe) {
_subscribe->setGlarePaused(paused);
}
};
_wrap.value(
) | rpl::start_with_next([=](Info::Wrap wrap) {
content->setRoundEdges(wrap == Info::Wrap::Layer);
}, content->lifetime());
content->setMaximumHeight(isEmojiStatus
? st::settingsPremiumUserHeight + TopTransitionSkip()
: st::settingsPremiumTopHeight);
content->setMinimumHeight(st::infoLayerTopBarHeight);
content->resize(content->width(), content->maximumHeight());
_wrap.value(
) | rpl::start_with_next([=](Info::Wrap wrap) {
const auto isLayer = (wrap == Info::Wrap::Layer);
_back = base::make_unique_q>(
content,
object_ptr(
content,
isEmojiStatus
? (isLayer ? st::infoTopBarBack : st::infoLayerTopBarBack)
: (isLayer
? st::settingsPremiumLayerTopBarBack
: st::settingsPremiumTopBarBack)),
st::infoTopBarScale);
_back->setDuration(0);
_back->toggleOn(isLayer
? _backToggles.value() | rpl::type_erased()
: rpl::single(true));
_back->entity()->addClickHandler([=] {
_showBack.fire({});
});
_back->toggledValue(
) | rpl::start_with_next([=](bool toggled) {
const auto &st = isLayer ? st::infoLayerTopBar : st::infoTopBar;
content->setTextPosition(
toggled ? st.back.width : st.titlePosition.x(),
st.titlePosition.y());
}, _back->lifetime());
if (!isLayer) {
_close = nullptr;
} else {
_close = base::make_unique_q(
content,
isEmojiStatus
? st::infoTopBarClose
: st::settingsPremiumTopBarClose);
_close->addClickHandler([=] {
_controller->parentController()->hideLayer();
_controller->parentController()->hideSpecialLayer();
});
content->widthValue(
) | rpl::start_with_next([=] {
_close->moveToRight(0, 0);
}, _close->lifetime());
}
}, content->lifetime());
return Ui::MakeWeak(not_null{ content });
}
void Premium::showFinished() {
_showFinished.fire({});
}
QPointer Premium::createPinnedToBottom(
not_null parent) {
const auto content = Ui::CreateChild(parent.get());
if (Ref::Gift::Parse(_ref)) {
return nullptr;
}
const auto emojiStatusData = Ref::EmojiStatus::Parse(_ref);
const auto session = &_controller->session();
auto buttonText = [&]() -> std::optional> {
if (emojiStatusData) {
auto &data = session->data();
if (const auto peer = data.peer(emojiStatusData.peerId)) {
return Info::Profile::EmojiStatusIdValue(
peer
) | rpl::map([=](DocumentId id) {
return id
? tr::lng_premium_emoji_status_button()
: _buttonText.value();
// : tr::lng_premium_summary_user_button();
}) | rpl::flatten_latest();
}
}
return _buttonText.value();
}();
_subscribe = CreateSubscribeButton({
_controller,
content,
[ref = _ref] { return ref; },
std::move(buttonText),
std::nullopt,
[=, options = session->api().premium().subscriptionOptions()] {
const auto value = _radioGroup->value();
return (value < options.size() && value >= 0)
? options[value].botUrl
: QString();
},
});
#if 0
if (emojiStatusData) {
// "Learn More" should open the general Premium Settings
// so we override the button callback.
// To have ability to jump back to the User Premium Settings
// we should replace the ref explicitly.
_subscribe->setClickedCallback([=] {
const auto ref = _ref;
const auto controller = _controller;
ShowPremium(controller, QString());
controller->setPremiumRef(ref);
});
} else {
#endif
{
_radioGroup->setChangedCallback([=](int value) {
const auto options =
_controller->session().api().premium().subscriptionOptions();
Expects(value < options.size() && value >= 0);
auto text = tr::lng_premium_subscribe_button(
tr::now,
lt_cost,
options[value].costPerMonth);
_buttonText = std::move(text);
});
_radioGroup->setValue(0);
}
_showFinished.events(
) | rpl::take(1) | rpl::start_with_next([=] {
_subscribe->startGlareAnimation();
}, _subscribe->lifetime());
content->widthValue(
) | rpl::start_with_next([=](int width) {
const auto padding = st::settingsPremiumButtonPadding;
_subscribe->resizeToWidth(width - padding.left() - padding.right());
}, _subscribe->lifetime());
rpl::combine(
_subscribe->heightValue(),
Data::AmPremiumValue(session),
session->premiumPossibleValue()
) | rpl::start_with_next([=](
int buttonHeight,
bool premium,
bool premiumPossible) {
const auto padding = st::settingsPremiumButtonPadding;
const auto finalHeight = !premiumPossible
? 0
: !premium
? (padding.top() + buttonHeight + padding.bottom())
: 0;
content->resize(content->width(), finalHeight);
_subscribe->moveToLeft(padding.left(), padding.top());
_subscribe->setVisible(!premium && premiumPossible);
}, _subscribe->lifetime());
return Ui::MakeWeak(not_null{ content });
}
} // namespace
Type PremiumId() {
return Premium::Id();
}
void ShowPremium(not_null session, const QString &ref) {
const auto active = Core::App().activeWindow();
const auto controller = (active && active->isPrimary())
? active->sessionController()
: nullptr;
if (controller && session == &controller->session()) {
ShowPremium(controller, ref);
} else {
for (const auto &controller : session->windows()) {
if (controller->window().isPrimary()) {
ShowPremium(controller, ref);
}
}
}
}
void ShowPremium(
not_null controller,
const QString &ref) {
if (!controller->session().premiumPossible()) {
controller->show(Box(PremiumUnavailableBox));
return;
}
controller->setPremiumRef(ref);
controller->showSettings(Settings::PremiumId());
}
void ShowGiftPremium(
not_null controller,
not_null peer,
int months,
bool me) {
ShowPremium(controller, Ref::Gift::Serialize({ peer->id, months, me }));
}
void ShowEmojiStatusPremium(
not_null controller,
not_null peer) {
ShowPremium(controller, Ref::EmojiStatus::Serialize({ peer->id }));
}
void StartPremiumPayment(
not_null controller,
const QString &ref) {
const auto account = &controller->session().account();
const auto username = account->appConfig().get(
"premium_bot_username",
QString());
const auto slug = account->appConfig().get(
"premium_invoice_slug",
QString());
if (!username.isEmpty()) {
controller->showPeerByLink(Window::SessionNavigation::PeerByLinkInfo{
.usernameOrId = username,
.resolveType = Window::ResolveType::BotStart,
.startToken = ref,
.startAutoSubmit = true,
});
} else if (!slug.isEmpty()) {
UrlClickHandler::Open("https://t.me/$" + slug);
}
}
QString LookupPremiumRef(PremiumPreview section) {
for (const auto &[ref, entry] : EntryMap()) {
if (entry.section == section) {
return ref;
}
}
return QString();
}
not_null CreateSubscribeButton(
SubscribeButtonArgs &&args) {
const auto result = Ui::CreateChild(
args.parent.get(),
args.gradientStops
? base::take(*args.gradientStops)
: Ui::Premium::ButtonGradientStops());
result->setClickedCallback([
controller = args.controller,
computeRef = args.computeRef,
computeBotUrl = args.computeBotUrl] {
const auto url = computeBotUrl ? QString() : computeBotUrl();
if (!url.isEmpty()) {
const auto local = Core::TryConvertUrlToLocal(url);
if (local.isEmpty()) {
return;
}
UrlClickHandler::Open(
local,
QVariant::fromValue(ClickHandlerContext{
.sessionWindow = base::make_weak(controller.get()),
.botStartAutoSubmit = true,
}));
} else {
SendScreenAccept(controller);
StartPremiumPayment(controller, computeRef());
}
});
const auto &st = st::premiumPreviewBox.button;
result->resize(args.parent->width(), st.height);
const auto premium = &args.controller->session().api().premium();
premium->reload();
const auto computeCost = [=] {
const auto amount = premium->monthlyAmount();
const auto currency = premium->monthlyCurrency();
const auto valid = (amount > 0) && !currency.isEmpty();
return Ui::FillAmountAndCurrency(
valid ? amount : 500,
valid ? currency : "USD");
};
const auto label = Ui::CreateChild(
result,
args.text
? base::take(*args.text)
: tr::lng_premium_summary_button(
lt_cost,
premium->statusTextValue() | rpl::map(computeCost)),
st::premiumPreviewButtonLabel);
label->setAttribute(Qt::WA_TransparentForMouseEvents);
rpl::combine(
result->widthValue(),
label->widthValue()
) | rpl::start_with_next([=](int outer, int width) {
label->moveToLeft(
(outer - width) / 2,
st::premiumPreviewBox.button.textTop,
outer);
}, label->lifetime());
return result;
}
} // namespace Settings