Update API scheme to layer 175. Business promo.
|
@ -1295,6 +1295,8 @@ PRIVATE
|
|||
settings/settings_advanced.h
|
||||
settings/settings_blocked_peers.cpp
|
||||
settings/settings_blocked_peers.h
|
||||
settings/settings_business.cpp
|
||||
settings/settings_business.h
|
||||
settings/settings_chat.cpp
|
||||
settings/settings_chat.h
|
||||
settings/settings_calls.cpp
|
||||
|
|
BIN
Telegram/Resources/art/business_logo.png
Normal file
After Width: | Height: | Size: 47 KiB |
BIN
Telegram/Resources/icons/menu/shop.png
Normal file
After Width: | Height: | Size: 687 B |
BIN
Telegram/Resources/icons/menu/shop@2x.png
Normal file
After Width: | Height: | Size: 1.2 KiB |
BIN
Telegram/Resources/icons/menu/shop@3x.png
Normal file
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 662 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.8 KiB |
After Width: | Height: | Size: 499 B |
After Width: | Height: | Size: 928 B |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 426 B |
After Width: | Height: | Size: 765 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 454 B |
After Width: | Height: | Size: 845 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 615 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.5 KiB |
Before Width: | Height: | Size: 497 B After Width: | Height: | Size: 568 B |
Before Width: | Height: | Size: 787 B After Width: | Height: | Size: 1.1 KiB |
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.6 KiB |
|
@ -2056,6 +2056,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
"lng_premium_summary_about_animated_userpics" = "Video avatars animated in chat lists and chats to allow for additional self-expression.";
|
||||
"lng_premium_summary_subtitle_translation" = "Real-Time Translation";
|
||||
"lng_premium_summary_about_translation" = "Real-time translation of channels and chats into other languages.";
|
||||
"lng_premium_summary_subtitle_business" = "Telegram Business";
|
||||
"lng_premium_summary_about_business" = "Upgrade your account with business features such as location, opening hours and quick replies.";
|
||||
"lng_premium_summary_bottom_subtitle" = "About Telegram Premium";
|
||||
"lng_premium_summary_bottom_about" = "While the free version of Telegram already gives its users more than any other messaging application, **Telegram Premium** pushes its capabilities even further.\n\n**Telegram Premium** is a paid option, because most Premium Features require additional expenses from Telegram to third parties such as data center providers and server manufacturers. Contributions from **Telegram Premium** users allow us to cover such costs and also help Telegram stay free for everyone.";
|
||||
"lng_premium_summary_button" = "Subscribe for {cost} per month";
|
||||
|
@ -2154,6 +2156,22 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
"lng_premium_gifts_terms" = "By gifting Telegram Premium, you agree to the Telegram {link} and {policy}.";
|
||||
"lng_premium_gifts_terms_policy" = "Privacy Policy";
|
||||
|
||||
"lng_business_title" = "Telegram Business";
|
||||
"lng_business_about" = "Turn your account to a business page with these additional features.";
|
||||
"lng_business_unlocked" = "You have now unlocked these additional business features.";
|
||||
"lng_business_subtitle_location" = "Location";
|
||||
"lng_business_about_location" = "Display the location of your business on your account.";
|
||||
"lng_business_subtitle_opening_hours" = "Opening Hours";
|
||||
"lng_business_about_opening_hours" = "Show to your customers when you are open for business.";
|
||||
"lng_business_subtitle_quick_replies" = "Quick Replies";
|
||||
"lng_business_about_quick_replies" = "Set up shortcuts up to 20 messages each to respond to customers faster.";
|
||||
"lng_business_subtitle_greeting_messages" = "Greeting Messages";
|
||||
"lng_business_about_greeting_messages" = "Create greetings that will be automatically sent to new customers.";
|
||||
"lng_business_subtitle_away_messages" = "Away Messages";
|
||||
"lng_business_about_away_messages" = "Define messages that are automatically sent when you are off.";
|
||||
"lng_business_subtitle_chatbots" = "Chatbots";
|
||||
"lng_business_about_chatbots" = "Add any third party chatbots that will process customer interactions.";
|
||||
|
||||
"lng_boost_channel_button" = "Boost Channel";
|
||||
"lng_boost_group_button" = "Boost Group";
|
||||
"lng_boost_again_button" = "Boost Again";
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
<file alias="art/background.tgv">../../art/background.tgv</file>
|
||||
<file alias="art/bg_thumbnail.png">../../art/bg_thumbnail.png</file>
|
||||
<file alias="art/bg_initial.jpg">../../art/bg_initial.jpg</file>
|
||||
<file alias="art/business_logo.png">../../art/business_logo.png</file>
|
||||
<file alias="art/logo_256.png">../../art/logo_256.png</file>
|
||||
<file alias="art/logo_256_no_margin.png">../../art/logo_256_no_margin.png</file>
|
||||
<file alias="art/themeimage.jpg">../../art/themeimage.jpg</file>
|
||||
|
|
|
@ -33,6 +33,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "ui/boxes/confirm_box.h"
|
||||
#include "ui/painter.h"
|
||||
#include "ui/vertical_list.h"
|
||||
#include "settings/settings_business.h"
|
||||
#include "settings/settings_premium.h"
|
||||
#include "lottie/lottie_single_player.h"
|
||||
#include "history/view/media/history_view_sticker.h"
|
||||
|
@ -128,6 +129,8 @@ void PreloadSticker(const std::shared_ptr<Data::DocumentMedia> &media) {
|
|||
return tr::lng_premium_summary_subtitle_animated_userpics();
|
||||
case PremiumPreview::RealTimeTranslation:
|
||||
return tr::lng_premium_summary_subtitle_translation();
|
||||
case PremiumPreview::Business:
|
||||
return tr::lng_premium_summary_subtitle_business();
|
||||
}
|
||||
Unexpected("PremiumPreview in SectionTitle.");
|
||||
}
|
||||
|
@ -170,6 +173,8 @@ void PreloadSticker(const std::shared_ptr<Data::DocumentMedia> &media) {
|
|||
return tr::lng_premium_summary_about_animated_userpics();
|
||||
case PremiumPreview::RealTimeTranslation:
|
||||
return tr::lng_premium_summary_about_translation();
|
||||
case PremiumPreview::Business:
|
||||
return tr::lng_premium_summary_about_business();
|
||||
}
|
||||
Unexpected("PremiumPreview in SectionTitle.");
|
||||
}
|
||||
|
@ -1219,6 +1224,13 @@ void Show(
|
|||
DecorateListPromoBox(box, show, descriptor);
|
||||
}));
|
||||
return;
|
||||
} else if (descriptor.section == PremiumPreview::Business) {
|
||||
const auto window = show->resolveWindow(
|
||||
ChatHelpers::WindowUsage::PremiumPromo);
|
||||
if (window) {
|
||||
Settings::ShowBusiness(window);
|
||||
}
|
||||
return;
|
||||
}
|
||||
auto &list = Preloads();
|
||||
for (auto i = begin(list); i != end(list);) {
|
||||
|
|
|
@ -64,6 +64,7 @@ enum class PremiumPreview {
|
|||
TagsForMessages,
|
||||
LastSeen,
|
||||
MessagePrivacy,
|
||||
Business,
|
||||
|
||||
kCount,
|
||||
};
|
||||
|
|
|
@ -402,6 +402,7 @@ updateBotMessageReactions#9cb7759 peer:Peer msg_id:int date:int reactions:Vector
|
|||
updateSavedDialogPinned#aeaf9e74 flags:# pinned:flags.0?true peer:DialogPeer = Update;
|
||||
updatePinnedSavedDialogs#686c85a6 flags:# order:flags.0?Vector<DialogPeer> = Update;
|
||||
updateSavedReactionTags#39c67432 = Update;
|
||||
updateSmsJob#f16269d4 job_id:string = Update;
|
||||
|
||||
updates.state#a56c2a3e pts:int qts:int date:int seq:int unread_count:int = updates.State;
|
||||
|
||||
|
@ -1653,6 +1654,12 @@ messages.savedReactionTags#3259950a tags:Vector<SavedReactionTag> hash:long = me
|
|||
|
||||
outboxReadDate#3bb842ac date:int = OutboxReadDate;
|
||||
|
||||
smsjobs.eligibleToJoin#dc8b44cf terms_url:string monthly_sent_sms:int = smsjobs.EligibilityToJoin;
|
||||
|
||||
smsjobs.status#2aee9191 flags:# allow_international:flags.0?true recent_sent:int recent_since:int recent_remains:int total_sent:int total_since:int last_gift_slug:flags.1?string terms_url:string = smsjobs.Status;
|
||||
|
||||
smsJob#e6a1eeb8 job_id:string phone_number:string text:string = SmsJob;
|
||||
|
||||
---functions---
|
||||
|
||||
invokeAfterMsg#cb9f372d {X:Type} msg_id:long query:!X = X;
|
||||
|
@ -2252,4 +2259,12 @@ premium.applyBoost#6b7da746 flags:# slots:flags.0?Vector<int> peer:InputPeer = p
|
|||
premium.getBoostsStatus#42f1f61 peer:InputPeer = premium.BoostsStatus;
|
||||
premium.getUserBoosts#39854d1f peer:InputPeer user_id:InputUser = premium.BoostsList;
|
||||
|
||||
// LAYER 174
|
||||
smsjobs.isEligibleToJoin#edc39d0 = smsjobs.EligibilityToJoin;
|
||||
smsjobs.join#a74ece2d = Bool;
|
||||
smsjobs.leave#9898ad73 = Bool;
|
||||
smsjobs.updateSettings#93fa0bf flags:# allow_international:flags.0?true = Bool;
|
||||
smsjobs.getStatus#10a698e8 = smsjobs.Status;
|
||||
smsjobs.getSmsJob#778d902f job_id:string = SmsJob;
|
||||
smsjobs.finishJob#4f1ebf24 flags:# job_id:string error:flags.0?string = Bool;
|
||||
|
||||
// LAYER 175
|
||||
|
|
|
@ -94,6 +94,7 @@ settingsPremiumIconTranslations: icon {{ "settings/premium/translations", settin
|
|||
settingsPremiumIconTags: icon {{ "settings/premium/tags", settingsIconFg }};
|
||||
settingsPremiumIconLastSeen: icon {{ "settings/premium/lastseen", settingsIconFg }};
|
||||
settingsPremiumIconPrivacy: icon {{ "settings/premium/privacy", settingsIconFg }};
|
||||
settingsPremiumIconBusiness: icon {{ "settings/premium/privacy", settingsIconFg }};
|
||||
|
||||
settingsStoriesIconOrder: icon {{ "settings/premium/stories_order", premiumButtonBg1 }};
|
||||
settingsStoriesIconStealth: icon {{ "menu/stealth", premiumButtonBg1 }};
|
||||
|
@ -103,6 +104,13 @@ settingsStoriesIconDownload: icon {{ "menu/download", premiumButtonBg1 }};
|
|||
settingsStoriesIconCaption: icon {{ "settings/premium/stories_caption", premiumButtonBg1 }};
|
||||
settingsStoriesIconLinks: icon {{ "menu/links_profile", premiumButtonBg1 }};
|
||||
|
||||
settingsBusinessIconLocation: icon {{ "settings/premium/business/business_location", settingsIconFg }};
|
||||
settingsBusinessIconHours: icon {{ "settings/premium/business/business_hours", settingsIconFg }};
|
||||
settingsBusinessIconReplies: icon {{ "settings/premium/business/business_quick", settingsIconFg }};
|
||||
settingsBusinessIconGreeting: icon {{ "settings/premium/status", settingsIconFg }};
|
||||
settingsBusinessIconAway: icon {{ "settings/premium/business/business_away", settingsIconFg }};
|
||||
settingsBusinessIconChatbots: icon {{ "settings/premium/business/business_chatbots", settingsIconFg }};
|
||||
|
||||
settingsPremiumNewBadge: FlatLabel(defaultFlatLabel) {
|
||||
style: TextStyle(semiboldTextStyle) {
|
||||
font: font(10px semibold);
|
||||
|
|
568
Telegram/SourceFiles/settings/settings_business.cpp
Normal file
|
@ -0,0 +1,568 @@
|
|||
/*
|
||||
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_business.h"
|
||||
|
||||
#include "boxes/premium_preview_box.h"
|
||||
#include "core/click_handler_types.h"
|
||||
#include "data/data_peer_values.h" // AmPremiumValue.
|
||||
#include "info/info_wrap_widget.h" // Info::Wrap.
|
||||
#include "info/settings/info_settings_widget.h" // SectionCustomTopBarData.
|
||||
#include "lang/lang_keys.h"
|
||||
#include "main/main_session.h"
|
||||
#include "settings/settings_common_session.h"
|
||||
#include "settings/settings_premium.h"
|
||||
#include "ui/effects/gradient.h"
|
||||
#include "ui/effects/premium_graphics.h"
|
||||
#include "ui/effects/premium_top_bar.h"
|
||||
#include "ui/layers/generic_box.h"
|
||||
#include "ui/text/text_utilities.h"
|
||||
#include "ui/widgets/checkbox.h" // Ui::RadiobuttonGroup.
|
||||
#include "ui/widgets/gradient_round_button.h"
|
||||
#include "ui/wrap/fade_wrap.h"
|
||||
#include "ui/wrap/slide_wrap.h"
|
||||
#include "ui/wrap/vertical_layout.h"
|
||||
#include "ui/vertical_list.h"
|
||||
#include "window/window_session_controller.h"
|
||||
#include "apiwrap.h"
|
||||
#include "api/api_premium.h"
|
||||
#include "styles/style_premium.h"
|
||||
#include "styles/style_info.h"
|
||||
#include "styles/style_layers.h"
|
||||
#include "styles/style_settings.h"
|
||||
|
||||
namespace Settings {
|
||||
namespace {
|
||||
|
||||
struct Entry {
|
||||
const style::icon *icon;
|
||||
rpl::producer<QString> title;
|
||||
rpl::producer<QString> description;
|
||||
BusinessFeature feature = BusinessFeature::Location;
|
||||
};
|
||||
|
||||
using Order = std::vector<QString>;
|
||||
|
||||
[[nodiscard]] Order FallbackOrder() {
|
||||
return Order{
|
||||
u"location"_q,
|
||||
u"opening_hours"_q,
|
||||
u"quick_replies"_q,
|
||||
u"greeting_messages"_q,
|
||||
u"away_messages"_q,
|
||||
u"chatbots"_q,
|
||||
};
|
||||
}
|
||||
|
||||
[[nodiscard]] base::flat_map<QString, Entry> EntryMap() {
|
||||
return base::flat_map<QString, Entry>{
|
||||
{
|
||||
u"location"_q,
|
||||
Entry{
|
||||
&st::settingsBusinessIconLocation,
|
||||
tr::lng_business_subtitle_location(),
|
||||
tr::lng_business_about_location(),
|
||||
BusinessFeature::Location,
|
||||
},
|
||||
},
|
||||
{
|
||||
u"opening_hours"_q,
|
||||
Entry{
|
||||
&st::settingsBusinessIconHours,
|
||||
tr::lng_business_subtitle_opening_hours(),
|
||||
tr::lng_business_about_opening_hours(),
|
||||
BusinessFeature::OpeningHours,
|
||||
},
|
||||
},
|
||||
{
|
||||
u"quick_replies"_q,
|
||||
Entry{
|
||||
&st::settingsBusinessIconReplies,
|
||||
tr::lng_business_subtitle_quick_replies(),
|
||||
tr::lng_business_about_quick_replies(),
|
||||
BusinessFeature::QuickReplies,
|
||||
},
|
||||
},
|
||||
{
|
||||
u"greeting_messages"_q,
|
||||
Entry{
|
||||
&st::settingsBusinessIconGreeting,
|
||||
tr::lng_business_subtitle_greeting_messages(),
|
||||
tr::lng_business_about_greeting_messages(),
|
||||
BusinessFeature::GreetingMessages,
|
||||
},
|
||||
},
|
||||
{
|
||||
u"away_messages"_q,
|
||||
Entry{
|
||||
&st::settingsBusinessIconAway,
|
||||
tr::lng_business_subtitle_away_messages(),
|
||||
tr::lng_business_about_away_messages(),
|
||||
BusinessFeature::AwayMessages,
|
||||
},
|
||||
},
|
||||
{
|
||||
u"chatbots"_q,
|
||||
Entry{
|
||||
&st::settingsBusinessIconChatbots,
|
||||
tr::lng_business_subtitle_chatbots(),
|
||||
tr::lng_business_about_chatbots(),
|
||||
BusinessFeature::Chatbots,
|
||||
},
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
void AddBusinessSummary(
|
||||
not_null<Ui::VerticalLayout*> content,
|
||||
not_null<Window::SessionController*> controller,
|
||||
Fn<void(BusinessFeature)> buttonCallback) {
|
||||
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;
|
||||
|
||||
auto entryMap = EntryMap();
|
||||
auto iconContainers = std::vector<Ui::AbstractButton*>();
|
||||
iconContainers.reserve(int(entryMap.size()));
|
||||
|
||||
const auto addRow = [&](Entry &entry) {
|
||||
const auto labelAscent = stLabel.style.font->ascent;
|
||||
const auto button = Ui::CreateChild<Ui::SettingsButton>(
|
||||
content.get(),
|
||||
rpl::single(QString()));
|
||||
|
||||
const auto label = content->add(
|
||||
object_ptr<Ui::FlatLabel>(
|
||||
content,
|
||||
std::move(entry.title) | rpl::map(Ui::Text::Bold),
|
||||
stLabel),
|
||||
titlePadding);
|
||||
label->setAttribute(Qt::WA_TransparentForMouseEvents);
|
||||
const auto description = content->add(
|
||||
object_ptr<Ui::FlatLabel>(
|
||||
content,
|
||||
std::move(entry.description),
|
||||
st::boxDividerLabel),
|
||||
descriptionPadding);
|
||||
description->setAttribute(Qt::WA_TransparentForMouseEvents);
|
||||
|
||||
const auto dummy = Ui::CreateChild<Ui::AbstractButton>(content.get());
|
||||
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<Ui::IconButton>(
|
||||
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 feature = entry.feature;
|
||||
button->setClickedCallback([=] { buttonCallback(feature); });
|
||||
|
||||
iconContainers.push_back(dummy);
|
||||
};
|
||||
|
||||
auto icons = std::vector<const style::icon *>();
|
||||
icons.reserve(int(entryMap.size()));
|
||||
{
|
||||
const auto &account = controller->session().account();
|
||||
const auto mtpOrder = FallbackOrder();/* session->account().appConfig().get<Order>(
|
||||
"premium_promo_order",
|
||||
FallbackOrder());*/ AssertIsDebug()
|
||||
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);
|
||||
}
|
||||
}
|
||||
|
||||
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 });
|
||||
}
|
||||
|
||||
Ui::AddSkip(content, descriptionPadding.bottom());
|
||||
}
|
||||
|
||||
class Business : public Section<Business> {
|
||||
public:
|
||||
Business(
|
||||
QWidget *parent,
|
||||
not_null<Window::SessionController*> controller);
|
||||
|
||||
[[nodiscard]] rpl::producer<QString> title() override;
|
||||
|
||||
[[nodiscard]] QPointer<Ui::RpWidget> createPinnedToTop(
|
||||
not_null<QWidget*> parent) override;
|
||||
[[nodiscard]] QPointer<Ui::RpWidget> createPinnedToBottom(
|
||||
not_null<Ui::RpWidget*> 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();
|
||||
|
||||
const not_null<Window::SessionController*> _controller;
|
||||
|
||||
QPointer<Ui::GradientButton> _subscribe;
|
||||
base::unique_qptr<Ui::FadeWrap<Ui::IconButton>> _back;
|
||||
base::unique_qptr<Ui::IconButton> _close;
|
||||
rpl::variable<bool> _backToggles;
|
||||
rpl::variable<Info::Wrap> _wrap;
|
||||
Fn<void(bool)> _setPaused;
|
||||
|
||||
std::shared_ptr<Ui::RadiobuttonGroup> _radioGroup;
|
||||
|
||||
rpl::event_stream<> _showBack;
|
||||
rpl::event_stream<> _showFinished;
|
||||
rpl::variable<QString> _buttonText;
|
||||
|
||||
};
|
||||
|
||||
Business::Business(
|
||||
QWidget *parent,
|
||||
not_null<Window::SessionController*> controller)
|
||||
: Section(parent)
|
||||
, _controller(controller)
|
||||
, _radioGroup(std::make_shared<Ui::RadiobuttonGroup>()) {
|
||||
setupContent();
|
||||
_controller->session().api().premium().reload();
|
||||
}
|
||||
|
||||
rpl::producer<QString> Business::title() {
|
||||
return tr::lng_premium_summary_title();
|
||||
}
|
||||
|
||||
bool Business::hasFlexibleTopBar() const {
|
||||
return true;
|
||||
}
|
||||
|
||||
rpl::producer<> Business::sectionShowBack() {
|
||||
return _showBack.events();
|
||||
}
|
||||
|
||||
void Business::setStepDataReference(std::any &data) {
|
||||
using namespace Info::Settings;
|
||||
const auto my = std::any_cast<SectionCustomTopBarData>(&data);
|
||||
if (my) {
|
||||
_backToggles = std::move(
|
||||
my->backButtonEnables
|
||||
) | rpl::map_to(true);
|
||||
_wrap = std::move(my->wrapValue);
|
||||
}
|
||||
}
|
||||
|
||||
void Business::setupContent() {
|
||||
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
|
||||
|
||||
Ui::AddSkip(content, st::settingsFromFileTop);
|
||||
|
||||
AddBusinessSummary(content, _controller, [=](BusinessFeature feature) {
|
||||
});
|
||||
|
||||
Ui::ResizeFitChild(this, content);
|
||||
}
|
||||
|
||||
QPointer<Ui::RpWidget> Business::createPinnedToTop(
|
||||
not_null<QWidget*> parent) {
|
||||
auto title = tr::lng_business_title();
|
||||
auto about = [&]() -> rpl::producer<TextWithEntities> {
|
||||
return rpl::conditional(
|
||||
Data::AmPremiumValue(&_controller->session()),
|
||||
tr::lng_business_unlocked(),
|
||||
tr::lng_business_about()
|
||||
) | Ui::Text::ToWithEntities();
|
||||
}();
|
||||
|
||||
const auto content = [&]() -> Ui::Premium::TopBarAbstract* {
|
||||
const auto weak = base::make_weak(_controller);
|
||||
const auto clickContextOther = [=] {
|
||||
return QVariant::fromValue(ClickHandlerContext{
|
||||
.sessionWindow = weak,
|
||||
.botStartAutoSubmit = true,
|
||||
});
|
||||
};
|
||||
return Ui::CreateChild<Ui::Premium::TopBar>(
|
||||
parent.get(),
|
||||
st::defaultPremiumCover,
|
||||
Ui::Premium::TopBarDescriptor{
|
||||
.clickContextOther = clickContextOther,
|
||||
.logo = u"dollar"_q,
|
||||
.title = std::move(title),
|
||||
.about = 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());
|
||||
|
||||
const auto calculateMaximumHeight = [=] {
|
||||
return st::settingsPremiumTopHeight;
|
||||
};
|
||||
|
||||
content->setMaximumHeight(calculateMaximumHeight());
|
||||
content->setMinimumHeight(st::settingsPremiumTopHeight);// st::infoLayerTopBarHeight);
|
||||
|
||||
content->resize(content->width(), content->maximumHeight());
|
||||
//content->additionalHeight(
|
||||
//) | rpl::start_with_next([=](int additionalHeight) {
|
||||
// const auto wasMax = (content->height() == content->maximumHeight());
|
||||
// content->setMaximumHeight(calculateMaximumHeight()
|
||||
// + additionalHeight);
|
||||
// if (wasMax) {
|
||||
// content->resize(content->width(), content->maximumHeight());
|
||||
// }
|
||||
//}, content->lifetime());
|
||||
|
||||
_wrap.value(
|
||||
) | rpl::start_with_next([=](Info::Wrap wrap) {
|
||||
const auto isLayer = (wrap == Info::Wrap::Layer);
|
||||
_back = base::make_unique_q<Ui::FadeWrap<Ui::IconButton>>(
|
||||
content,
|
||||
object_ptr<Ui::IconButton>(
|
||||
content,
|
||||
(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<Ui::IconButton>(
|
||||
content,
|
||||
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<Ui::RpWidget*>{ content });
|
||||
}
|
||||
|
||||
void Business::showFinished() {
|
||||
_showFinished.fire({});
|
||||
}
|
||||
|
||||
QPointer<Ui::RpWidget> Business::createPinnedToBottom(
|
||||
not_null<Ui::RpWidget*> parent) {
|
||||
const auto content = Ui::CreateChild<Ui::RpWidget>(parent.get());
|
||||
|
||||
const auto session = &_controller->session();
|
||||
|
||||
auto buttonText = _buttonText.value();
|
||||
|
||||
_subscribe = CreateSubscribeButton({
|
||||
_controller,
|
||||
content,
|
||||
[] { return u"business"_q; },
|
||||
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();
|
||||
},
|
||||
});
|
||||
{
|
||||
const auto callback = [=](int value) {
|
||||
const auto options =
|
||||
_controller->session().api().premium().subscriptionOptions();
|
||||
if (options.empty()) {
|
||||
return;
|
||||
}
|
||||
Assert(value < options.size() && value >= 0);
|
||||
auto text = tr::lng_premium_subscribe_button(
|
||||
tr::now,
|
||||
lt_cost,
|
||||
options[value].costPerMonth);
|
||||
_buttonText = std::move(text);
|
||||
};
|
||||
_radioGroup->setChangedCallback(callback);
|
||||
callback(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<Ui::RpWidget*>{ content });
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
template <>
|
||||
struct SectionFactory<Business> : AbstractSectionFactory {
|
||||
object_ptr<AbstractSection> create(
|
||||
not_null<QWidget*> parent,
|
||||
not_null<Window::SessionController*> controller
|
||||
) const final override {
|
||||
return object_ptr<Business>(parent, controller);
|
||||
}
|
||||
bool hasCustomTopBar() const final override {
|
||||
return true;
|
||||
}
|
||||
|
||||
[[nodiscard]] static const std::shared_ptr<SectionFactory> &Instance() {
|
||||
static const auto result = std::make_shared<SectionFactory>();
|
||||
return result;
|
||||
}
|
||||
};
|
||||
|
||||
Type BusinessId() {
|
||||
return Business::Id();
|
||||
}
|
||||
|
||||
void ShowBusiness(not_null<Window::SessionController*> controller) {
|
||||
if (!controller->session().premiumPossible()) {
|
||||
controller->show(Box(PremiumUnavailableBox));
|
||||
return;
|
||||
}
|
||||
controller->showSettings(Settings::BusinessId());
|
||||
}
|
||||
|
||||
} // namespace Settings
|
37
Telegram/SourceFiles/settings/settings_business.h
Normal file
|
@ -0,0 +1,37 @@
|
|||
/*
|
||||
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
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#include "settings/settings_type.h"
|
||||
|
||||
namespace Main {
|
||||
class Session;
|
||||
} // namespace Main
|
||||
|
||||
namespace Window {
|
||||
class SessionController;
|
||||
} // namespace Window
|
||||
|
||||
namespace Settings {
|
||||
|
||||
enum class BusinessFeature {
|
||||
Location,
|
||||
OpeningHours,
|
||||
QuickReplies,
|
||||
GreetingMessages,
|
||||
AwayMessages,
|
||||
Chatbots,
|
||||
|
||||
kCount,
|
||||
};
|
||||
|
||||
[[nodiscard]] Type BusinessId();
|
||||
|
||||
void ShowBusiness(not_null<Window::SessionController*> controller);
|
||||
|
||||
} // namespace Settings
|
|
@ -7,6 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
*/
|
||||
#include "settings/settings_main.h"
|
||||
|
||||
#include "settings/settings_business.h"
|
||||
#include "settings/settings_codes.h"
|
||||
#include "settings/settings_chat.h"
|
||||
#include "settings/settings_information.h"
|
||||
|
@ -419,6 +420,19 @@ void SetupPremium(
|
|||
controller->setPremiumRef("settings");
|
||||
showOther(PremiumId());
|
||||
});
|
||||
const auto button = AddButtonWithIcon(
|
||||
container,
|
||||
tr::lng_business_title(),
|
||||
st::settingsButton,
|
||||
{ .icon = &st::menuIconShop });
|
||||
button->addClickHandler([=] {
|
||||
showOther(BusinessId());
|
||||
});
|
||||
constexpr auto kNewExpiresAt = int(1711958400);
|
||||
if (base::unixtime::now() < kNewExpiresAt) {
|
||||
Ui::NewBadge::AddToRight(button);
|
||||
}
|
||||
|
||||
if (controller->session().premiumCanBuy()) {
|
||||
const auto button = AddButtonWithIcon(
|
||||
container,
|
||||
|
|
|
@ -180,6 +180,7 @@ using Order = std::vector<QString>;
|
|||
u"stories"_q,
|
||||
u"more_upload"_q,
|
||||
u"double_limits"_q,
|
||||
u"business"_q,
|
||||
u"last_seen"_q,
|
||||
u"voice_to_text"_q,
|
||||
u"faster_download"_q,
|
||||
|
@ -367,6 +368,16 @@ using Order = std::vector<QString>;
|
|||
PremiumPreview::RealTimeTranslation,
|
||||
},
|
||||
},
|
||||
{
|
||||
u"business"_q,
|
||||
Entry{
|
||||
&st::settingsPremiumIconPlay, AssertIsDebug()
|
||||
tr::lng_premium_summary_subtitle_business(),
|
||||
tr::lng_premium_summary_about_business(),
|
||||
PremiumPreview::Business,
|
||||
true,
|
||||
},
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -1671,9 +1682,9 @@ void AddSummaryPremium(
|
|||
icons.reserve(int(entryMap.size()));
|
||||
{
|
||||
const auto &account = controller->session().account();
|
||||
const auto mtpOrder = account.appConfig().get<Order>(
|
||||
const auto mtpOrder = FallbackOrder();/* session->account().appConfig().get<Order>(
|
||||
"premium_promo_order",
|
||||
FallbackOrder());
|
||||
FallbackOrder());*/ AssertIsDebug()
|
||||
const auto processEntry = [&](Entry &entry) {
|
||||
icons.push_back(entry.icon);
|
||||
addRow(entry);
|
||||
|
|
|
@ -25,6 +25,21 @@ constexpr auto kBodyAnimationPart = 0.90;
|
|||
constexpr auto kTitleAdditionalScale = 0.15;
|
||||
constexpr auto kMinAcceptableContrast = 4.5; // 1.14;
|
||||
|
||||
[[nodiscard]] QImage ScaleTo(QImage image) {
|
||||
using namespace style;
|
||||
const auto size = image.size();
|
||||
const auto scale = DevicePixelRatio() * Scale() / 300.;
|
||||
const auto scaled = QSize(
|
||||
int(base::SafeRound(size.width() * scale)),
|
||||
int(base::SafeRound(size.height() * scale)));
|
||||
image = image.scaled(
|
||||
scaled,
|
||||
Qt::IgnoreAspectRatio,
|
||||
Qt::SmoothTransformation);
|
||||
image.setDevicePixelRatio(DevicePixelRatio());
|
||||
return image;
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
QString Svg() {
|
||||
|
@ -157,27 +172,41 @@ TopBar::TopBar(
|
|||
rpl::producer<TextWithEntities> about,
|
||||
bool light,
|
||||
bool optimizeMinistars)
|
||||
: TopBar(parent, st, {
|
||||
.clickContextOther = std::move(clickContextOther),
|
||||
.title = std::move(title),
|
||||
.about = std::move(about),
|
||||
.light = light,
|
||||
.optimizeMinistars = optimizeMinistars,
|
||||
}) {
|
||||
}
|
||||
|
||||
TopBar::TopBar(
|
||||
not_null<QWidget*> parent,
|
||||
const style::PremiumCover &st,
|
||||
TopBarDescriptor &&descriptor)
|
||||
: TopBarAbstract(parent, st)
|
||||
, _light(light)
|
||||
, _light(descriptor.light)
|
||||
, _logo(descriptor.logo)
|
||||
, _titleFont(st.titleFont)
|
||||
, _titlePadding(st.titlePadding)
|
||||
, _about(this, std::move(about), st.about)
|
||||
, _ministars(this, optimizeMinistars) {
|
||||
, _about(this, std::move(descriptor.about), st.about)
|
||||
, _ministars(this, descriptor.optimizeMinistars) {
|
||||
std::move(
|
||||
title
|
||||
descriptor.title
|
||||
) | rpl::start_with_next([=](QString text) {
|
||||
_titlePath = QPainterPath();
|
||||
_titlePath.addText(0, _titleFont->ascent, _titleFont, text);
|
||||
update();
|
||||
}, lifetime());
|
||||
|
||||
if (clickContextOther) {
|
||||
if (const auto other = descriptor.clickContextOther) {
|
||||
_about->setClickHandlerFilter([=](
|
||||
const ClickHandlerPtr &handler,
|
||||
Qt::MouseButton button) {
|
||||
ActivateClickHandler(_about, handler, {
|
||||
button,
|
||||
clickContextOther()
|
||||
other()
|
||||
});
|
||||
return false;
|
||||
});
|
||||
|
@ -188,7 +217,10 @@ TopBar::TopBar(
|
|||
) | rpl::start_with_next([=] {
|
||||
TopBarAbstract::computeIsDark();
|
||||
|
||||
if (!_light && !TopBarAbstract::isDark()) {
|
||||
if (_logo == u"dollar"_q) {
|
||||
_dollar = ScaleTo(QImage(u":/gui/art/business_logo.png"_q));
|
||||
_ministars.setColorOverride(st::premiumButtonFg->c);
|
||||
} else if (!_light && !TopBarAbstract::isDark()) {
|
||||
_star.load(Svg());
|
||||
_ministars.setColorOverride(st::premiumButtonFg->c);
|
||||
} else {
|
||||
|
@ -232,8 +264,11 @@ rpl::producer<int> TopBar::additionalHeight() const {
|
|||
}
|
||||
|
||||
void TopBar::resizeEvent(QResizeEvent *e) {
|
||||
const auto progress = (e->size().height() - minimumHeight())
|
||||
/ float64(maximumHeight() - minimumHeight());
|
||||
const auto max = maximumHeight();
|
||||
const auto min = minimumHeight();
|
||||
const auto progress = (max > min)
|
||||
? ((e->size().height() - min) / float64(max - min))
|
||||
: 1.;
|
||||
_progress.top = 1. -
|
||||
std::clamp(
|
||||
(1. - progress) / kBodyAnimationPart,
|
||||
|
@ -291,7 +326,12 @@ void TopBar::paintEvent(QPaintEvent *e) {
|
|||
}
|
||||
p.resetTransform();
|
||||
|
||||
_star.render(&p, _starRect);
|
||||
if (!_dollar.isNull()) {
|
||||
auto hq = PainterHighQualityEnabler(p);
|
||||
p.drawImage(_starRect, _dollar);
|
||||
} else {
|
||||
_star.render(&p, _starRect);
|
||||
}
|
||||
|
||||
const auto color = _light
|
||||
? st::settingsPremiumUserTitle.textFg
|
||||
|
|
|
@ -64,6 +64,15 @@ private:
|
|||
|
||||
};
|
||||
|
||||
struct TopBarDescriptor {
|
||||
Fn<QVariant()> clickContextOther;
|
||||
QString logo;
|
||||
rpl::producer<QString> title;
|
||||
rpl::producer<TextWithEntities> about;
|
||||
bool light = false;
|
||||
bool optimizeMinistars = true;
|
||||
};
|
||||
|
||||
class TopBar final : public TopBarAbstract {
|
||||
public:
|
||||
TopBar(
|
||||
|
@ -74,6 +83,10 @@ public:
|
|||
rpl::producer<TextWithEntities> about,
|
||||
bool light = false,
|
||||
bool optimizeMinistars = true);
|
||||
TopBar(
|
||||
not_null<QWidget*> parent,
|
||||
const style::PremiumCover &st,
|
||||
TopBarDescriptor &&descriptor);
|
||||
~TopBar();
|
||||
|
||||
void setPaused(bool paused) override;
|
||||
|
@ -87,11 +100,13 @@ protected:
|
|||
|
||||
private:
|
||||
const bool _light = false;
|
||||
const QString _logo;
|
||||
const style::font &_titleFont;
|
||||
const style::margins &_titlePadding;
|
||||
object_ptr<FlatLabel> _about;
|
||||
ColoredMiniStars _ministars;
|
||||
QSvgRenderer _star;
|
||||
QImage _dollar;
|
||||
|
||||
struct {
|
||||
float64 top = 0.;
|
||||
|
|
|
@ -137,6 +137,7 @@ menuIconAntispam: icon {{ "menu/antispam", menuIconColor }};
|
|||
menuIconChatDiscuss: icon {{ "menu/chat_discuss", menuIconColor }};
|
||||
menuIconBotCommands: icon {{ "menu/bot_commands", menuIconColor }};
|
||||
menuIconPremium: icon {{ "menu/premium", menuIconColor }};
|
||||
menuIconShop: icon {{ "menu/shop", menuIconColor }};
|
||||
menuIconIpAddress: icon {{ "menu/ip_address", menuIconColor }};
|
||||
menuIconAddress: icon {{ "menu/payment_address", menuIconColor }};
|
||||
menuIconShowAll: icon {{ "menu/all_media", menuIconColor }};
|
||||
|
|