mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-07-19 04:02:50 +02:00
767 lines
22 KiB
C++
767 lines
22 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 "info/bot/earn/info_earn_inner_widget.h"
|
|
|
|
#include "api/api_credits.h"
|
|
#include "api/api_earn.h"
|
|
#include "api/api_filter_updates.h"
|
|
#include "base/timer_rpl.h"
|
|
#include "base/unixtime.h"
|
|
#include "core/ui_integration.h"
|
|
#include "data/data_channel_earn.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_user.h"
|
|
#include "data/stickers/data_custom_emoji.h"
|
|
#include "info/bot/earn/info_earn_widget.h"
|
|
#include "info/channel_statistics/earn/earn_format.h"
|
|
#include "info/info_controller.h"
|
|
#include "info/statistics/info_statistics_inner_widget.h" // FillLoading.
|
|
#include "info/statistics/info_statistics_list_controllers.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "main/main_account.h"
|
|
#include "main/main_app_config.h"
|
|
#include "main/main_session.h"
|
|
#include "settings/settings_credits_graphics.h"
|
|
#include "statistics/chart_widget.h"
|
|
#include "statistics/widgets/chart_header_widget.h"
|
|
#include "ui/effects/credits_graphics.h"
|
|
#include "ui/layers/generic_box.h"
|
|
#include "ui/rect.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/toast/toast.h"
|
|
#include "ui/vertical_list.h"
|
|
#include "ui/widgets/buttons.h"
|
|
#include "ui/widgets/discrete_sliders.h"
|
|
#include "ui/widgets/fields/number_input.h"
|
|
#include "ui/widgets/label_with_custom_emoji.h"
|
|
#include "ui/widgets/labels.h"
|
|
#include "ui/wrap/slide_wrap.h"
|
|
#include "styles/style_boxes.h"
|
|
#include "styles/style_channel_earn.h"
|
|
#include "styles/style_chat.h"
|
|
#include "styles/style_credits.h"
|
|
#include "styles/style_layers.h"
|
|
#include "styles/style_settings.h"
|
|
#include "styles/style_statistics.h"
|
|
|
|
namespace Info::BotEarn {
|
|
namespace {
|
|
|
|
[[nodiscard]] int WithdrawalMin(not_null<Main::Session*> session) {
|
|
const auto key = u"stars_revenue_withdrawal_min"_q;
|
|
return session->appConfig().get<int>(key, 1000);
|
|
}
|
|
|
|
void AddHeader(
|
|
not_null<Ui::VerticalLayout*> content,
|
|
tr::phrase<> text) {
|
|
Ui::AddSkip(content);
|
|
const auto header = content->add(
|
|
object_ptr<Ui::FlatLabel>(
|
|
content,
|
|
text(),
|
|
st::channelEarnHeaderLabel),
|
|
st::boxRowPadding);
|
|
header->resizeToWidth(header->width());
|
|
}
|
|
|
|
[[nodiscard]] not_null<Ui::RpWidget*> CreateIconWidget(
|
|
not_null<Ui::RpWidget*> parent,
|
|
QImage image) {
|
|
const auto widget = Ui::CreateChild<Ui::RpWidget>(parent);
|
|
widget->resize(image.size() / style::DevicePixelRatio());
|
|
widget->paintRequest(
|
|
) | rpl::start_with_next([=] {
|
|
auto p = QPainter(widget);
|
|
p.drawImage(0, 0, image);
|
|
}, widget->lifetime());
|
|
widget->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
return widget;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
InnerWidget::InnerWidget(
|
|
QWidget *parent,
|
|
not_null<Controller*> controller,
|
|
not_null<PeerData*> peer)
|
|
: VerticalLayout(parent)
|
|
, _controller(controller)
|
|
, _peer(peer)
|
|
, _show(controller->uiShow()) {
|
|
}
|
|
|
|
void InnerWidget::load() {
|
|
const auto apiLifetime = lifetime().make_state<rpl::lifetime>();
|
|
|
|
const auto request = [=](Fn<void(Data::BotEarnStatistics)> done) {
|
|
const auto api = apiLifetime->make_state<Api::BotEarnStatistics>(
|
|
_peer->asUser());
|
|
api->request(
|
|
) | rpl::start_with_error_done([show = _show](const QString &error) {
|
|
show->showToast(error);
|
|
}, [=] {
|
|
done(api->data());
|
|
apiLifetime->destroy();
|
|
}, *apiLifetime);
|
|
};
|
|
|
|
Info::Statistics::FillLoading(
|
|
this,
|
|
_loaded.events_starting_with(false) | rpl::map(!rpl::mappers::_1),
|
|
_showFinished.events());
|
|
|
|
_showFinished.events(
|
|
) | rpl::take(1) | rpl::start_with_next([=] {
|
|
request([=](Data::BotEarnStatistics state) {
|
|
_state = state;
|
|
_loaded.fire(true);
|
|
fill();
|
|
|
|
_peer->session().account().mtpUpdates(
|
|
) | rpl::start_with_next([=](const MTPUpdates &updates) {
|
|
using TL = MTPDupdateStarsRevenueStatus;
|
|
Api::PerformForUpdate<TL>(updates, [&](const TL &d) {
|
|
const auto peerId = peerFromMTP(d.vpeer());
|
|
if (peerId == _peer->id) {
|
|
request([=](Data::BotEarnStatistics state) {
|
|
_state = state;
|
|
_stateUpdated.fire({});
|
|
});
|
|
}
|
|
});
|
|
}, lifetime());
|
|
});
|
|
}, lifetime());
|
|
}
|
|
|
|
void InnerWidget::fill() {
|
|
using namespace Info::ChannelEarn;
|
|
const auto container = this;
|
|
const auto &data = _state;
|
|
const auto multiplier = data.usdRate * Data::kEarnMultiplier;
|
|
const auto session = &_peer->session();
|
|
|
|
auto availableBalanceValue = rpl::single(
|
|
data.availableBalance
|
|
) | rpl::then(
|
|
_stateUpdated.events() | rpl::map([=] {
|
|
return _state.availableBalance;
|
|
})
|
|
);
|
|
auto valueToString = [](uint64 v) { return QString::number(v); };
|
|
|
|
if (data.revenueGraph.chart) {
|
|
Ui::AddSkip(container);
|
|
Ui::AddSkip(container);
|
|
using Type = Statistic::ChartViewType;
|
|
const auto widget = container->add(
|
|
object_ptr<Statistic::ChartWidget>(container),
|
|
st::statisticsLayerMargins);
|
|
|
|
auto chart = data.revenueGraph.chart;
|
|
chart.currencyRate = data.usdRate;
|
|
|
|
widget->setChartData(chart, Type::StackBar);
|
|
widget->setTitle(tr::lng_bot_earn_chart_revenue());
|
|
Ui::AddSkip(container);
|
|
Ui::AddDivider(container);
|
|
Ui::AddSkip(container);
|
|
Ui::AddSkip(container);
|
|
}
|
|
{
|
|
AddHeader(container, tr::lng_bot_earn_overview_title);
|
|
Ui::AddSkip(container, st::channelEarnOverviewTitleSkip);
|
|
|
|
const auto addOverview = [&](
|
|
rpl::producer<uint64> value,
|
|
const tr::phrase<> &text) {
|
|
const auto line = container->add(
|
|
Ui::CreateSkipWidget(container, 0),
|
|
st::boxRowPadding);
|
|
const auto majorLabel = Ui::CreateChild<Ui::FlatLabel>(
|
|
line,
|
|
rpl::duplicate(value) | rpl::map(valueToString),
|
|
st::channelEarnOverviewMajorLabel);
|
|
const auto icon = CreateIconWidget(
|
|
line,
|
|
Ui::GenerateStars(majorLabel->height(), 1));
|
|
const auto secondMinorLabel = Ui::CreateChild<Ui::FlatLabel>(
|
|
line,
|
|
std::move(
|
|
value
|
|
) | rpl::map([=](uint64 v) {
|
|
return v ? ToUsd(v, multiplier) : QString();
|
|
}),
|
|
st::channelEarnOverviewSubMinorLabel);
|
|
rpl::combine(
|
|
line->widthValue(),
|
|
majorLabel->sizeValue()
|
|
) | rpl::start_with_next([=](int available, const QSize &size) {
|
|
line->resize(line->width(), size.height());
|
|
majorLabel->moveToLeft(
|
|
icon->width() + st::channelEarnOverviewMinorLabelSkip,
|
|
majorLabel->y());
|
|
secondMinorLabel->resizeToWidth(available
|
|
- size.width()
|
|
- icon->width());
|
|
secondMinorLabel->moveToLeft(
|
|
rect::right(majorLabel)
|
|
+ st::channelEarnOverviewSubMinorLabelPos.x(),
|
|
st::channelEarnOverviewSubMinorLabelPos.y());
|
|
}, majorLabel->lifetime());
|
|
Ui::ToggleChildrenVisibility(line, true);
|
|
|
|
Ui::AddSkip(container);
|
|
const auto sub = container->add(
|
|
object_ptr<Ui::FlatLabel>(
|
|
container,
|
|
text(),
|
|
st::channelEarnOverviewSubMinorLabel),
|
|
st::boxRowPadding);
|
|
};
|
|
addOverview(
|
|
rpl::duplicate(availableBalanceValue),
|
|
tr::lng_bot_earn_available);
|
|
Ui::AddSkip(container);
|
|
Ui::AddSkip(container);
|
|
// addOverview(data.currentBalance, tr::lng_bot_earn_reward);
|
|
// Ui::AddSkip(container);
|
|
// Ui::AddSkip(container);
|
|
addOverview(
|
|
rpl::single(
|
|
data.overallRevenue
|
|
) | rpl::then(
|
|
_stateUpdated.events() | rpl::map([=] {
|
|
return _state.overallRevenue;
|
|
})
|
|
),
|
|
tr::lng_bot_earn_total);
|
|
Ui::AddSkip(container);
|
|
Ui::AddSkip(container);
|
|
Ui::AddDividerText(container, tr::lng_bot_earn_balance_about());
|
|
Ui::AddSkip(container);
|
|
}
|
|
{
|
|
AddHeader(container, tr::lng_bot_earn_balance_title);
|
|
Ui::AddSkip(container);
|
|
|
|
const auto labels = container->add(
|
|
object_ptr<Ui::CenterWrap<Ui::RpWidget>>(
|
|
container,
|
|
object_ptr<Ui::RpWidget>(container)))->entity();
|
|
|
|
const auto majorLabel = Ui::CreateChild<Ui::FlatLabel>(
|
|
labels,
|
|
rpl::duplicate(availableBalanceValue) | rpl::map(valueToString),
|
|
st::channelEarnBalanceMajorLabel);
|
|
const auto icon = CreateIconWidget(
|
|
labels,
|
|
Ui::GenerateStars(majorLabel->height(), 1));
|
|
majorLabel->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
majorLabel->sizeValue(
|
|
) | rpl::start_with_next([=](const QSize &majorSize) {
|
|
const auto skip = st::channelEarnBalanceMinorLabelSkip;
|
|
labels->resize(
|
|
majorSize.width() + icon->width() + skip,
|
|
majorSize.height());
|
|
majorLabel->moveToLeft(icon->width() + skip, 0);
|
|
}, labels->lifetime());
|
|
Ui::ToggleChildrenVisibility(labels, true);
|
|
|
|
Ui::AddSkip(container);
|
|
container->add(
|
|
object_ptr<Ui::CenterWrap<>>(
|
|
container,
|
|
object_ptr<Ui::FlatLabel>(
|
|
container,
|
|
rpl::duplicate(
|
|
availableBalanceValue
|
|
) | rpl::map([=](uint64 v) {
|
|
return v ? ToUsd(v, multiplier) : QString();
|
|
}),
|
|
st::channelEarnOverviewSubMinorLabel)));
|
|
|
|
Ui::AddSkip(container);
|
|
|
|
const auto input = [&] {
|
|
const auto &st = st::botEarnInputField;
|
|
const auto inputContainer = container->add(
|
|
Ui::CreateSkipWidget(container, st.heightMin));
|
|
const auto currentValue = rpl::variable<uint64>(
|
|
rpl::duplicate(availableBalanceValue));
|
|
const auto input = Ui::CreateChild<Ui::NumberInput>(
|
|
inputContainer,
|
|
st,
|
|
tr::lng_bot_earn_out_ph(),
|
|
QString::number(currentValue.current()),
|
|
currentValue.current());
|
|
rpl::duplicate(
|
|
availableBalanceValue
|
|
) | rpl::start_with_next([=](uint64 v) {
|
|
input->changeLimit(v);
|
|
input->setText(QString::number(v));
|
|
}, input->lifetime());
|
|
const auto icon = CreateIconWidget(
|
|
inputContainer,
|
|
Ui::GenerateStars(st.style.font->height, 1));
|
|
inputContainer->sizeValue(
|
|
) | rpl::start_with_next([=](const QSize &size) {
|
|
input->resize(
|
|
size.width() - rect::m::sum::h(st::boxRowPadding),
|
|
st.heightMin);
|
|
input->moveToLeft(st::boxRowPadding.left(), 0);
|
|
icon->moveToLeft(
|
|
st::boxRowPadding.left(),
|
|
st.textMargins.top());
|
|
}, input->lifetime());
|
|
Ui::ToggleChildrenVisibility(inputContainer, true);
|
|
return input;
|
|
}();
|
|
|
|
Ui::AddSkip(container);
|
|
Ui::AddSkip(container);
|
|
|
|
auto dateValue = rpl::single(
|
|
data.nextWithdrawalAt
|
|
) | rpl::then(
|
|
_stateUpdated.events() | rpl::map([=] {
|
|
return _state.nextWithdrawalAt;
|
|
})
|
|
);
|
|
auto lockedValue = rpl::duplicate(
|
|
dateValue
|
|
) | rpl::map([=](const QDateTime &dt) {
|
|
return !dt.isNull() || (!_state.isWithdrawalEnabled);
|
|
});
|
|
|
|
const auto &stButton = st::defaultActiveButton;
|
|
const auto button = container->add(
|
|
object_ptr<Ui::RoundButton>(
|
|
container,
|
|
rpl::never<QString>(),
|
|
stButton),
|
|
st::boxRowPadding);
|
|
|
|
rpl::duplicate(
|
|
lockedValue
|
|
) | rpl::start_with_next([=](bool v) {
|
|
button->setAttribute(Qt::WA_TransparentForMouseEvents, v);
|
|
}, button->lifetime());
|
|
|
|
const auto label = Ui::CreateChild<Ui::FlatLabel>(
|
|
button,
|
|
tr::lng_channel_earn_balance_button(tr::now),
|
|
st::channelEarnSemiboldLabel);
|
|
const auto processInputChange = [&] {
|
|
const auto buttonEmoji = Ui::Text::SingleCustomEmoji(
|
|
session->data().customEmojiManager().registerInternalEmoji(
|
|
st::settingsPremiumIconStar,
|
|
{ 0, -st::moderateBoxExpandInnerSkip, 0, 0 },
|
|
true));
|
|
const auto context = Core::MarkedTextContext{
|
|
.session = session,
|
|
.customEmojiRepaint = [=] { label->update(); },
|
|
};
|
|
const auto process = [=] {
|
|
const auto amount = input->getLastText().toDouble();
|
|
if (amount >= _state.availableBalance) {
|
|
label->setText(
|
|
tr::lng_bot_earn_balance_button_all(tr::now));
|
|
} else {
|
|
label->setMarkedText(
|
|
tr::lng_bot_earn_balance_button(
|
|
tr::now,
|
|
lt_count,
|
|
amount,
|
|
lt_emoji,
|
|
buttonEmoji,
|
|
Ui::Text::RichLangValue),
|
|
context);
|
|
}
|
|
};
|
|
QObject::connect(input, &Ui::MaskedInputField::changed, process);
|
|
process();
|
|
return process;
|
|
}();
|
|
label->setTextColorOverride(stButton.textFg->c);
|
|
label->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
rpl::combine(
|
|
rpl::duplicate(lockedValue),
|
|
button->sizeValue(),
|
|
label->sizeValue()
|
|
) | rpl::start_with_next([=](bool v, const QSize &b, const QSize &l) {
|
|
label->moveToLeft(
|
|
(b.width() - l.width()) / 2,
|
|
(v ? -10 : 1) * (b.height() - l.height()) / 2);
|
|
}, label->lifetime());
|
|
|
|
const auto lockedColor = anim::with_alpha(stButton.textFg->c, .5);
|
|
const auto lockedLabelTop = Ui::CreateChild<Ui::FlatLabel>(
|
|
button,
|
|
tr::lng_bot_earn_balance_button_locked(),
|
|
st::botEarnLockedButtonLabel);
|
|
lockedLabelTop->setTextColorOverride(lockedColor);
|
|
lockedLabelTop->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
const auto lockedLabelBottom = Ui::CreateChild<Ui::FlatLabel>(
|
|
button,
|
|
QString(),
|
|
st::botEarnLockedButtonLabel);
|
|
lockedLabelBottom->setTextColorOverride(lockedColor);
|
|
lockedLabelBottom->setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
rpl::combine(
|
|
rpl::duplicate(lockedValue),
|
|
button->sizeValue(),
|
|
lockedLabelTop->sizeValue(),
|
|
lockedLabelBottom->sizeValue()
|
|
) | rpl::start_with_next([=](
|
|
bool locked,
|
|
const QSize &b,
|
|
const QSize &top,
|
|
const QSize &bottom) {
|
|
const auto factor = locked ? 1 : -10;
|
|
const auto sumHeight = top.height() + bottom.height();
|
|
lockedLabelTop->moveToLeft(
|
|
(b.width() - top.width()) / 2,
|
|
factor * (b.height() - sumHeight) / 2);
|
|
lockedLabelBottom->moveToLeft(
|
|
(b.width() - bottom.width()) / 2,
|
|
factor * ((b.height() - sumHeight) / 2 + top.height()));
|
|
}, lockedLabelTop->lifetime());
|
|
|
|
const auto dateUpdateLifetime
|
|
= lockedLabelBottom->lifetime().make_state<rpl::lifetime>();
|
|
std::move(
|
|
dateValue
|
|
) | rpl::start_with_next([=](const QDateTime &dt) {
|
|
dateUpdateLifetime->destroy();
|
|
if (dt.isNull()) {
|
|
return;
|
|
}
|
|
constexpr auto kDateUpdateInterval = crl::time(250);
|
|
const auto was = base::unixtime::serialize(dt);
|
|
|
|
const auto context = Core::MarkedTextContext{
|
|
.session = session,
|
|
.customEmojiRepaint = [=] { lockedLabelBottom->update(); },
|
|
};
|
|
const auto emoji = Ui::Text::SingleCustomEmoji(
|
|
session->data().customEmojiManager().registerInternalEmoji(
|
|
st::chatSimilarLockedIcon,
|
|
st::botEarnButtonLockMargins,
|
|
true));
|
|
|
|
rpl::single(
|
|
rpl::empty
|
|
) | rpl::then(
|
|
base::timer_each(kDateUpdateInterval)
|
|
) | rpl::start_with_next([=] {
|
|
const auto secondsDifference = std::max(
|
|
was - base::unixtime::now() - 1,
|
|
0);
|
|
const auto hours = secondsDifference / 3600;
|
|
const auto minutes = (secondsDifference % 3600) / 60;
|
|
const auto seconds = secondsDifference % 60;
|
|
constexpr auto kZero = QChar('0');
|
|
const auto formatted = (hours > 0)
|
|
? (u"%1:%2:%3"_q)
|
|
.arg(hours, 2, 10, kZero)
|
|
.arg(minutes, 2, 10, kZero)
|
|
.arg(seconds, 2, 10, kZero)
|
|
: (u"%1:%2"_q)
|
|
.arg(minutes, 2, 10, kZero)
|
|
.arg(seconds, 2, 10, kZero);
|
|
lockedLabelBottom->setMarkedText(
|
|
base::duplicate(emoji).append(formatted),
|
|
context);
|
|
}, *dateUpdateLifetime);
|
|
}, lockedLabelBottom->lifetime());
|
|
|
|
Api::HandleWithdrawalButton(
|
|
Api::RewardReceiver{
|
|
.creditsReceiver = _peer,
|
|
.creditsAmount = [=, show = _controller->uiShow()] {
|
|
const auto amount = input->getLastText().toULongLong();
|
|
const auto min = float64(WithdrawalMin(session));
|
|
if (amount <= min) {
|
|
auto text = tr::lng_bot_earn_credits_out_minimal(
|
|
tr::now,
|
|
lt_link,
|
|
Ui::Text::Link(
|
|
tr::lng_bot_earn_credits_out_minimal_link(
|
|
tr::now,
|
|
lt_count,
|
|
min),
|
|
u"internal:"_q),
|
|
Ui::Text::RichLangValue);
|
|
show->showToast(Ui::Toast::Config{
|
|
.text = std::move(text),
|
|
.filter = [=](const auto ...) {
|
|
input->setText(QString::number(min));
|
|
processInputChange();
|
|
return true;
|
|
},
|
|
});
|
|
return 0ULL;
|
|
}
|
|
return amount;
|
|
},
|
|
},
|
|
button,
|
|
_controller->uiShow());
|
|
Ui::ToggleChildrenVisibility(button, true);
|
|
|
|
Ui::AddSkip(container);
|
|
Ui::AddSkip(container);
|
|
|
|
const auto arrow = Ui::Text::SingleCustomEmoji(
|
|
session->data().customEmojiManager().registerInternalEmoji(
|
|
st::topicButtonArrow,
|
|
st::channelEarnLearnArrowMargins,
|
|
false));
|
|
auto about = Ui::CreateLabelWithCustomEmoji(
|
|
container,
|
|
tr::lng_bot_earn_learn_credits_out_about(
|
|
lt_link,
|
|
tr::lng_channel_earn_about_link(
|
|
lt_emoji,
|
|
rpl::single(arrow),
|
|
Ui::Text::RichLangValue
|
|
) | rpl::map([](TextWithEntities text) {
|
|
return Ui::Text::Link(
|
|
std::move(text),
|
|
tr::lng_bot_earn_balance_about_url(tr::now));
|
|
}),
|
|
Ui::Text::RichLangValue),
|
|
{ .session = session },
|
|
st::boxDividerLabel);
|
|
Ui::AddSkip(container);
|
|
container->add(object_ptr<Ui::DividerLabel>(
|
|
container,
|
|
std::move(about),
|
|
st::defaultBoxDividerLabelPadding,
|
|
RectPart::Top | RectPart::Bottom));
|
|
|
|
Ui::AddSkip(container);
|
|
}
|
|
|
|
fillHistory();
|
|
}
|
|
|
|
void InnerWidget::fillHistory() {
|
|
const auto container = this;
|
|
const auto history = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(container)));
|
|
const auto content = history->entity();
|
|
|
|
Ui::AddSkip(content, st::settingsPremiumOptionsPadding.top());
|
|
|
|
const auto fill = [=](
|
|
not_null<PeerData*> premiumBot,
|
|
const Data::CreditsStatusSlice &fullSlice,
|
|
const Data::CreditsStatusSlice &inSlice,
|
|
const Data::CreditsStatusSlice &outSlice) {
|
|
const auto inner = content;
|
|
if (fullSlice.list.empty()) {
|
|
return;
|
|
}
|
|
const auto hasOneTab = inSlice.list.empty() && outSlice.list.empty();
|
|
const auto hasIn = !inSlice.list.empty();
|
|
const auto hasOut = !outSlice.list.empty();
|
|
const auto fullTabText = tr::lng_credits_summary_history_tab_full(
|
|
tr::now);
|
|
const auto inTabText = tr::lng_credits_summary_history_tab_in(
|
|
tr::now);
|
|
const auto outTabText = tr::lng_credits_summary_history_tab_out(
|
|
tr::now);
|
|
if (hasOneTab) {
|
|
Ui::AddSkip(inner);
|
|
const auto header = inner->add(
|
|
object_ptr<Statistic::Header>(inner),
|
|
st::statisticsLayerMargins
|
|
+ st::boostsChartHeaderPadding);
|
|
header->resizeToWidth(header->width());
|
|
header->setTitle(fullTabText);
|
|
header->setSubTitle({});
|
|
}
|
|
|
|
class Slider final : public Ui::SettingsSlider {
|
|
public:
|
|
using Ui::SettingsSlider::SettingsSlider;
|
|
void setNaturalWidth(int w) {
|
|
_naturalWidth = w;
|
|
}
|
|
int naturalWidth() const override {
|
|
return _naturalWidth;
|
|
}
|
|
|
|
private:
|
|
int _naturalWidth = 0;
|
|
|
|
};
|
|
|
|
const auto slider = inner->add(
|
|
object_ptr<Ui::SlideWrap<Slider>>(
|
|
inner,
|
|
object_ptr<Slider>(inner, st::defaultTabsSlider)),
|
|
st::boxRowPadding);
|
|
slider->toggle(!hasOneTab, anim::type::instant);
|
|
|
|
slider->entity()->addSection(fullTabText);
|
|
if (hasIn) {
|
|
slider->entity()->addSection(inTabText);
|
|
}
|
|
if (hasOut) {
|
|
slider->entity()->addSection(outTabText);
|
|
}
|
|
|
|
{
|
|
const auto &st = st::defaultTabsSlider;
|
|
slider->entity()->setNaturalWidth(0
|
|
+ st.labelStyle.font->width(fullTabText)
|
|
+ (hasIn ? st.labelStyle.font->width(inTabText) : 0)
|
|
+ (hasOut ? st.labelStyle.font->width(outTabText) : 0)
|
|
+ rect::m::sum::h(st::boxRowPadding));
|
|
}
|
|
|
|
const auto fullWrap = inner->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
inner,
|
|
object_ptr<Ui::VerticalLayout>(inner)));
|
|
const auto inWrap = inner->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
inner,
|
|
object_ptr<Ui::VerticalLayout>(inner)));
|
|
const auto outWrap = inner->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
inner,
|
|
object_ptr<Ui::VerticalLayout>(inner)));
|
|
|
|
rpl::single(0) | rpl::then(
|
|
slider->entity()->sectionActivated()
|
|
) | rpl::start_with_next([=](int index) {
|
|
if (index == 0) {
|
|
fullWrap->toggle(true, anim::type::instant);
|
|
inWrap->toggle(false, anim::type::instant);
|
|
outWrap->toggle(false, anim::type::instant);
|
|
} else if (index == 1) {
|
|
inWrap->toggle(true, anim::type::instant);
|
|
fullWrap->toggle(false, anim::type::instant);
|
|
outWrap->toggle(false, anim::type::instant);
|
|
} else {
|
|
outWrap->toggle(true, anim::type::instant);
|
|
fullWrap->toggle(false, anim::type::instant);
|
|
inWrap->toggle(false, anim::type::instant);
|
|
}
|
|
}, inner->lifetime());
|
|
|
|
const auto controller = _controller->parentController();
|
|
const auto entryClicked = [=](const Data::CreditsHistoryEntry &e) {
|
|
controller->uiShow()->show(Box(
|
|
::Settings::ReceiptCreditsBox,
|
|
controller,
|
|
premiumBot.get(),
|
|
e));
|
|
};
|
|
|
|
const auto star = lifetime().make_state<QImage>(
|
|
Ui::GenerateStars(st::creditsTopupButton.height, 1));
|
|
|
|
Info::Statistics::AddCreditsHistoryList(
|
|
controller->uiShow(),
|
|
fullSlice,
|
|
fullWrap->entity(),
|
|
entryClicked,
|
|
premiumBot,
|
|
star,
|
|
true,
|
|
true);
|
|
Info::Statistics::AddCreditsHistoryList(
|
|
controller->uiShow(),
|
|
inSlice,
|
|
inWrap->entity(),
|
|
entryClicked,
|
|
premiumBot,
|
|
star,
|
|
true,
|
|
false);
|
|
Info::Statistics::AddCreditsHistoryList(
|
|
controller->uiShow(),
|
|
outSlice,
|
|
outWrap->entity(),
|
|
std::move(entryClicked),
|
|
premiumBot,
|
|
star,
|
|
false,
|
|
true);
|
|
|
|
Ui::AddSkip(inner);
|
|
Ui::AddSkip(inner);
|
|
|
|
inner->resizeToWidth(container->width());
|
|
};
|
|
|
|
const auto apiLifetime = content->lifetime().make_state<rpl::lifetime>();
|
|
{
|
|
using Api = Api::CreditsHistory;
|
|
const auto apiFull = apiLifetime->make_state<Api>(_peer, true, true);
|
|
const auto apiIn = apiLifetime->make_state<Api>(_peer, true, false);
|
|
const auto apiOut = apiLifetime->make_state<Api>(_peer, false, true);
|
|
apiFull->request({}, [=](Data::CreditsStatusSlice fullSlice) {
|
|
apiIn->request({}, [=](Data::CreditsStatusSlice inSlice) {
|
|
apiOut->request({}, [=](Data::CreditsStatusSlice outSlice) {
|
|
::Api::PremiumPeerBot(
|
|
&_controller->session()
|
|
) | rpl::start_with_next([=](not_null<PeerData*> bot) {
|
|
fill(bot, fullSlice, inSlice, outSlice);
|
|
apiLifetime->destroy();
|
|
}, *apiLifetime);
|
|
});
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
void InnerWidget::saveState(not_null<Memento*> memento) {
|
|
memento->setState(base::take(_state));
|
|
}
|
|
|
|
void InnerWidget::restoreState(not_null<Memento*> memento) {
|
|
_state = memento->state();
|
|
if (_state) {
|
|
fill();
|
|
} else {
|
|
load();
|
|
}
|
|
Ui::RpWidget::resizeToWidth(width());
|
|
}
|
|
|
|
rpl::producer<Ui::ScrollToRequest> InnerWidget::scrollToRequests() const {
|
|
return _scrollToRequests.events();
|
|
}
|
|
|
|
auto InnerWidget::showRequests() const -> rpl::producer<ShowRequest> {
|
|
return _showRequests.events();
|
|
}
|
|
|
|
void InnerWidget::showFinished() {
|
|
_showFinished.fire({});
|
|
}
|
|
|
|
void InnerWidget::setInnerFocus() {
|
|
_focusRequested.fire({});
|
|
}
|
|
|
|
not_null<PeerData*> InnerWidget::peer() const {
|
|
return _peer;
|
|
}
|
|
|
|
} // namespace Info::BotEarn
|
|
|