| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128 |
- /*
- 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_privacy_security.h"
- #include "api/api_authorizations.h"
- #include "api/api_cloud_password.h"
- #include "api/api_self_destruct.h"
- #include "api/api_sensitive_content.h"
- #include "api/api_global_privacy.h"
- #include "api/api_websites.h"
- #include "settings/cloud_password/settings_cloud_password_email_confirm.h"
- #include "settings/cloud_password/settings_cloud_password_input.h"
- #include "settings/cloud_password/settings_cloud_password_start.h"
- #include "settings/settings_active_sessions.h"
- #include "settings/settings_blocked_peers.h"
- #include "settings/settings_global_ttl.h"
- #include "settings/settings_local_passcode.h"
- #include "settings/settings_premium.h" // Settings::ShowPremium.
- #include "settings/settings_privacy_controllers.h"
- #include "settings/settings_websites.h"
- #include "base/system_unlock.h"
- #include "base/timer_rpl.h"
- #include "boxes/passcode_box.h"
- #include "ui/boxes/confirm_box.h"
- #include "boxes/self_destruction_box.h"
- #include "core/application.h"
- #include "core/core_settings.h"
- #include "ui/chat/chat_style.h"
- #include "ui/effects/premium_graphics.h"
- #include "ui/effects/premium_top_bar.h"
- #include "ui/text/format_values.h"
- #include "ui/text/text_utilities.h"
- #include "ui/toast/toast.h"
- #include "ui/wrap/slide_wrap.h"
- #include "ui/wrap/fade_wrap.h"
- #include "ui/widgets/fields/input_field.h"
- #include "ui/widgets/shadow.h"
- #include "ui/widgets/checkbox.h"
- #include "ui/vertical_list.h"
- #include "ui/rect.h"
- #include "calls/calls_instance.h"
- #include "core/update_checker.h"
- #include "lang/lang_keys.h"
- #include "data/components/top_peers.h"
- #include "data/data_session.h"
- #include "data/data_chat.h"
- #include "data/data_channel.h"
- #include "data/data_peer_values.h"
- #include "main/main_app_config.h"
- #include "main/main_domain.h"
- #include "main/main_session.h"
- #include "storage/storage_domain.h"
- #include "window/window_session_controller.h"
- #include "apiwrap.h"
- #include "styles/style_settings.h"
- #include "styles/style_menu_icons.h"
- #include "styles/style_layers.h"
- #include <QtGui/QGuiApplication>
- #include <QtSvg/QSvgRenderer>
- namespace Settings {
- namespace {
- constexpr auto kUpdateTimeout = 60 * crl::time(1000);
- using Privacy = Api::UserPrivacy;
- [[nodiscard]] QImage PremiumStar() {
- const auto factor = style::DevicePixelRatio();
- const auto size = Size(st::settingsButtonNoIcon.style.font->ascent);
- auto image = QImage(
- size * factor,
- QImage::Format_ARGB32_Premultiplied);
- image.setDevicePixelRatio(factor);
- image.fill(Qt::transparent);
- {
- auto p = QPainter(&image);
- auto star = QSvgRenderer(
- Ui::Premium::ColorizedSvg(Ui::Premium::ButtonGradientStops()));
- star.render(&p, Rect(size));
- }
- return image;
- }
- void AddPremiumStar(
- not_null<Ui::SettingsButton*> button,
- not_null<Main::Session*> session,
- rpl::producer<QString> label,
- const QMargins &padding) {
- const auto badge = Ui::CreateChild<Ui::RpWidget>(button.get());
- badge->showOn(Data::AmPremiumValue(session));
- const auto sampleLeft = st::settingsColorSamplePadding.left();
- const auto badgeLeft = padding.left() + sampleLeft;
- auto star = PremiumStar();
- badge->resize(star.size() / style::DevicePixelRatio());
- badge->paintRequest(
- ) | rpl::start_with_next([=] {
- auto p = QPainter(badge);
- p.drawImage(0, 0, star);
- }, badge->lifetime());
- rpl::combine(
- button->sizeValue(),
- std::move(label)
- ) | rpl::start_with_next([=](const QSize &s, const QString &) {
- if (s.isNull()) {
- return;
- }
- badge->moveToLeft(
- button->fullTextWidth() + badgeLeft,
- (s.height() - badge->height()) / 2);
- }, badge->lifetime());
- }
- void OpenFileConfirmationsBox(not_null<Ui::GenericBox*> box) {
- box->setTitle(tr::lng_settings_file_confirmations());
- const auto settings = &Core::App().settings();
- const auto &list = settings->noWarningExtensions();
- const auto text = QStringList(begin(list), end(list)).join(' ');
- const auto layout = box->verticalLayout();
- const auto extensions = box->addRow(
- object_ptr<Ui::InputField>(
- box,
- st::defaultInputField,
- Ui::InputField::Mode::MultiLine,
- tr::lng_settings_edit_extensions(),
- TextWithTags{ text }),
- st::boxRowPadding + QMargins(0, 0, 0, st::settingsPrivacySkip));
- Ui::AddDividerText(layout, tr::lng_settings_edit_extensions_about());
- Ui::AddSkip(layout);
- const auto ip = layout->add(object_ptr<Ui::SettingsButton>(
- box,
- tr::lng_settings_edit_ip_confirm(),
- st::settingsButtonNoIcon
- ))->toggleOn(rpl::single(settings->ipRevealWarning()));
- Ui::AddSkip(layout);
- Ui::AddDividerText(layout, tr::lng_settings_edit_ip_confirm_about());
- box->setFocusCallback([=] {
- extensions->setFocusFast();
- });
- box->addButton(tr::lng_settings_save(), [=] {
- const auto extensionsList = extensions->getLastText()
- .mid(0, 10240)
- .split(' ', Qt::SkipEmptyParts)
- .mid(0, 1024);
- auto extensions = base::flat_set<QString>(
- extensionsList.begin(),
- extensionsList.end());
- const auto ipRevealWarning = ip->toggled();
- if (extensions != settings->noWarningExtensions()
- || ipRevealWarning != settings->ipRevealWarning()) {
- settings->setNoWarningExtensions(std::move(extensions));
- settings->setIpRevealWarning(ipRevealWarning);
- Core::App().saveSettingsDelayed();
- }
- box->closeBox();
- });
- box->addButton(tr::lng_cancel(), [=] { box->closeBox(); });
- }
- QString PrivacyBase(Privacy::Key key, const Privacy::Rule &rule) {
- using Key = Privacy::Key;
- using Option = Privacy::Option;
- switch (key) {
- case Key::CallsPeer2Peer:
- switch (rule.option) {
- case Option::Everyone:
- return tr::lng_edit_privacy_calls_p2p_everyone(tr::now);
- case Option::Contacts:
- return tr::lng_edit_privacy_calls_p2p_contacts(tr::now);
- case Option::Nobody:
- return tr::lng_edit_privacy_calls_p2p_nobody(tr::now);
- }
- [[fallthrough]];
- default:
- switch (rule.option) {
- case Option::Everyone:
- return rule.never.miniapps
- ? tr::lng_edit_privacy_no_miniapps(tr::now)
- : tr::lng_edit_privacy_everyone(tr::now);
- case Option::Contacts:
- return rule.always.premiums
- ? tr::lng_edit_privacy_contacts_and_premium(tr::now)
- : rule.always.miniapps
- ? tr::lng_edit_privacy_contacts_and_miniapps(tr::now)
- : tr::lng_edit_privacy_contacts(tr::now);
- case Option::CloseFriends:
- return tr::lng_edit_privacy_close_friends(tr::now);
- case Option::Nobody:
- return rule.always.premiums
- ? tr::lng_edit_privacy_premium(tr::now)
- : rule.always.miniapps
- ? tr::lng_edit_privacy_miniapps(tr::now)
- : tr::lng_edit_privacy_nobody(tr::now);
- }
- Unexpected("Value in Privacy::Option.");
- }
- }
- rpl::producer<QString> PrivacyString(
- not_null<::Main::Session*> session,
- Privacy::Key key) {
- session->api().userPrivacy().reload(key);
- return session->api().userPrivacy().value(
- key
- ) | rpl::map([=](const Privacy::Rule &value) {
- auto add = QStringList();
- if (const auto never = ExceptionUsersCount(value.never.peers)) {
- add.push_back("-" + QString::number(never));
- }
- if (const auto always = ExceptionUsersCount(value.always.peers)) {
- add.push_back("+" + QString::number(always));
- }
- if (!add.isEmpty()) {
- return PrivacyBase(key, value)
- + " (" + add.join(", ") + ")";
- } else {
- return PrivacyBase(key, value);
- }
- });
- }
- #if 0 // Dead code.
- void AddPremiumPrivacyButton(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<QString> label,
- Privacy::Key key,
- Fn<std::unique_ptr<EditPrivacyController>()> controllerFactory) {
- const auto shower = Ui::CreateChild<rpl::lifetime>(container.get());
- const auto session = &controller->session();
- const auto &st = st::settingsButtonNoIcon;
- const auto button = container->add(object_ptr<Button>(
- container,
- rpl::duplicate(label),
- st));
- AddPremiumStar(button, session, rpl::duplicate(label), st.padding);
- struct State {
- State(QWidget *parent) : widget(parent) {
- widget.setAttribute(Qt::WA_TransparentForMouseEvents);
- }
- Ui::RpWidget widget;
- };
- const auto state = button->lifetime().make_state<State>(button);
- using WeakToast = base::weak_ptr<Ui::Toast::Instance>;
- const auto toast = std::make_shared<WeakToast>();
- {
- const auto rightLabel = Ui::CreateChild<Ui::FlatLabel>(
- button,
- st.rightLabel);
- state->widget.resize(st::settingsPremiumLock.size());
- state->widget.paintRequest(
- ) | rpl::filter([=]() -> bool {
- return state->widget.x();
- }) | rpl::start_with_next([=] {
- auto p = QPainter(&state->widget);
- st::settingsPremiumLock.paint(p, 0, 0, state->widget.width());
- }, state->widget.lifetime());
- rpl::combine(
- button->sizeValue(),
- std::move(label),
- PrivacyString(session, key),
- Data::AmPremiumValue(session)
- ) | rpl::start_with_next([=, &st](
- const QSize &buttonSize,
- const QString &button,
- const QString &text,
- bool premium) {
- const auto locked = !premium;
- const auto rightSkip = st::settingsButtonRightSkip;
- const auto lockSkip = st::settingsPremiumLockSkip;
- const auto available = buttonSize.width()
- - st.padding.left()
- - st.padding.right()
- - st.style.font->width(button)
- - rightSkip
- - (locked ? state->widget.width() + lockSkip : 0);
- rightLabel->setText(text);
- rightLabel->resizeToNaturalWidth(available);
- rightLabel->moveToRight(
- rightSkip,
- st.padding.top());
- state->widget.moveToRight(
- rightSkip + rightLabel->width() + lockSkip,
- (buttonSize.height() - state->widget.height()) / 2);
- state->widget.setVisible(locked);
- }, rightLabel->lifetime());
- rightLabel->setAttribute(Qt::WA_TransparentForMouseEvents);
- }
- const auto showToast = [=] {
- auto link = Ui::Text::Link(
- Ui::Text::Semibold(
- tr::lng_settings_privacy_premium_link(tr::now)));
- (*toast) = controller->showToast({
- .text = tr::lng_settings_privacy_premium(
- tr::now,
- lt_link,
- link,
- Ui::Text::WithEntities),
- .duration = Ui::Toast::kDefaultDuration * 2,
- .filter = crl::guard(&controller->session(), [=](
- const ClickHandlerPtr &,
- Qt::MouseButton button) {
- if (button == Qt::LeftButton) {
- if (const auto strong = toast->get()) {
- strong->hideAnimated();
- (*toast) = nullptr;
- Settings::ShowPremium(controller, QString());
- return true;
- }
- }
- return false;
- }),
- });
- };
- button->addClickHandler([=] {
- if (!session->premium()) {
- if (toast->empty()) {
- showToast();
- }
- return;
- }
- *shower = session->api().userPrivacy().value(
- key
- ) | rpl::take(
- 1
- ) | rpl::start_with_next([=](const Privacy::Rule &value) {
- controller->show(Box<EditPrivacyBox>(
- controller,
- controllerFactory(),
- value));
- });
- });
- }
- #endif
- void AddMessagesPrivacyButton(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container) {
- const auto session = &controller->session();
- const auto privacy = &session->api().globalPrivacy();
- auto label = rpl::combine(
- privacy->newRequirePremium(),
- privacy->newChargeStars()
- ) | rpl::map([=](bool requirePremium, int chargeStars) {
- return chargeStars
- ? tr::lng_edit_privacy_paid()
- : requirePremium
- ? tr::lng_edit_privacy_contacts_and_premium()
- : tr::lng_edit_privacy_everyone();
- }) | rpl::flatten_latest();
- const auto &st = st::settingsButtonNoIcon;
- const auto button = AddButtonWithLabel(
- container,
- tr::lng_settings_messages_privacy(),
- rpl::duplicate(label),
- st,
- {});
- button->addClickHandler([=] {
- controller->show(Box(EditMessagesPrivacyBox, controller));
- });
- if (!session->appConfig().newRequirePremiumFree()) {
- AddPremiumStar(button, session, rpl::duplicate(label), st.padding);
- }
- }
- rpl::producer<int> BlockedPeersCount(not_null<::Main::Session*> session) {
- return session->api().blockedPeers().slice(
- ) | rpl::map([](const Api::BlockedPeers::Slice &data) {
- return data.total;
- });
- }
- void SetupPrivacy(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger) {
- Ui::AddSkip(container, st::settingsPrivacySkip);
- Ui::AddSubsectionTitle(container, tr::lng_settings_privacy_title());
- const auto session = &controller->session();
- using Key = Privacy::Key;
- const auto add = [&](
- rpl::producer<QString> label,
- Key key,
- auto controllerFactory) {
- return AddPrivacyButton(
- controller,
- container,
- std::move(label),
- {},
- key,
- controllerFactory);
- };
- add(
- tr::lng_settings_phone_number_privacy(),
- Key::PhoneNumber,
- [=] { return std::make_unique<PhoneNumberPrivacyController>(
- controller); });
- add(
- tr::lng_settings_last_seen(),
- Key::LastSeen,
- [=] { return std::make_unique<LastSeenPrivacyController>(
- session); });
- add(
- tr::lng_settings_profile_photo_privacy(),
- Key::ProfilePhoto,
- [] { return std::make_unique<ProfilePhotoPrivacyController>(); });
- add(
- tr::lng_settings_bio_privacy(),
- Key::About,
- [] { return std::make_unique<AboutPrivacyController>(); });
- add(
- tr::lng_settings_gifts_privacy(),
- Key::GiftsAutoSave,
- [=] { return std::make_unique<GiftsAutoSavePrivacyController>(); });
- add(
- tr::lng_settings_birthday_privacy(),
- Key::Birthday,
- [] { return std::make_unique<BirthdayPrivacyController>(); });
- add(
- tr::lng_settings_forwards_privacy(),
- Key::Forwards,
- [=] { return std::make_unique<ForwardsPrivacyController>(
- controller); });
- add(
- tr::lng_settings_calls(),
- Key::Calls,
- [] { return std::make_unique<CallsPrivacyController>(); });
- add(
- tr::lng_settings_groups_invite(),
- Key::Invites,
- [] { return std::make_unique<GroupsInvitePrivacyController>(); });
- {
- const auto &phrase = tr::lng_settings_voices_privacy;
- const auto &st = st::settingsButtonNoIcon;
- auto callback = [=] {
- return std::make_unique<VoicesPrivacyController>(session);
- };
- const auto voices = add(phrase(), Key::Voices, std::move(callback));
- AddPremiumStar(voices, session, phrase(), st.padding);
- }
- AddMessagesPrivacyButton(controller, container);
- session->api().userPrivacy().reload(
- Api::UserPrivacy::Key::AddedByPhone);
- Ui::AddSkip(container, st::settingsPrivacySecurityPadding);
- Ui::AddDivider(container);
- }
- void SetupLocalPasscode(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- Fn<void(Type)> showOther) {
- auto has = rpl::single(rpl::empty) | rpl::then(
- controller->session().domain().local().localPasscodeChanged()
- ) | rpl::map([=] {
- return controller->session().domain().local().hasLocalPasscode();
- });
- auto label = rpl::combine(
- tr::lng_settings_cloud_password_on(),
- tr::lng_settings_cloud_password_off(),
- std::move(has),
- [](const QString &on, const QString &off, bool has) {
- return has ? on : off;
- });
- AddButtonWithLabel(
- container,
- tr::lng_settings_passcode_title(),
- std::move(label),
- st::settingsButton,
- { &st::menuIconLock }
- )->addClickHandler([=] {
- if (controller->session().domain().local().hasLocalPasscode()) {
- showOther(LocalPasscodeCheckId());
- } else {
- showOther(LocalPasscodeCreateId());
- }
- });
- }
- void SetupCloudPassword(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- Fn<void(Type)> showOther) {
- using namespace rpl::mappers;
- using State = Core::CloudPasswordState;
- enum class PasswordState {
- Loading,
- On,
- Off,
- Unconfirmed,
- };
- const auto session = &controller->session();
- auto passwordState = rpl::single(
- PasswordState::Loading
- ) | rpl::then(session->api().cloudPassword().state(
- ) | rpl::map([](const State &state) {
- return (!state.unconfirmedPattern.isEmpty())
- ? PasswordState::Unconfirmed
- : state.hasPassword
- ? PasswordState::On
- : PasswordState::Off;
- })) | rpl::distinct_until_changed();
- auto label = rpl::duplicate(
- passwordState
- ) | rpl::map([=](PasswordState state) {
- return (state == PasswordState::Loading)
- ? tr::lng_profile_loading(tr::now)
- : (state == PasswordState::On)
- ? tr::lng_settings_cloud_password_on(tr::now)
- : tr::lng_settings_cloud_password_off(tr::now);
- });
- AddButtonWithLabel(
- container,
- tr::lng_settings_cloud_password_start_title(),
- std::move(label),
- st::settingsButton,
- { &st::menuIconPermissions }
- )->addClickHandler([=, passwordState = base::duplicate(passwordState)] {
- const auto state = rpl::variable<PasswordState>(
- base::duplicate(passwordState)).current();
- if (state == PasswordState::Loading) {
- return;
- } else if (state == PasswordState::On) {
- showOther(CloudPasswordInputId());
- } else if (state == PasswordState::Off) {
- showOther(CloudPasswordStartId());
- } else if (state == PasswordState::Unconfirmed) {
- showOther(CloudPasswordEmailConfirmId());
- }
- });
- const auto reloadOnActivation = [=](Qt::ApplicationState state) {
- if (/*label->toggled() && */state == Qt::ApplicationActive) {
- controller->session().api().cloudPassword().reload();
- }
- };
- QObject::connect(
- static_cast<QGuiApplication*>(QCoreApplication::instance()),
- &QGuiApplication::applicationStateChanged,
- container,
- reloadOnActivation);
- session->api().cloudPassword().reload();
- }
- void SetupTopPeers(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container) {
- Ui::AddSkip(container);
- Ui::AddSubsectionTitle(container, tr::lng_settings_top_peers_title());
- const auto session = &controller->session();
- container->add(object_ptr<Button>(
- container,
- tr::lng_settings_top_peers_suggest(),
- st::settingsButtonNoIcon
- ))->toggleOn(rpl::single(
- rpl::empty
- ) | rpl::then(
- session->topPeers().updates()
- ) | rpl::map([=] {
- return !session->topPeers().disabled();
- }))->toggledChanges(
- ) | rpl::filter([=](bool enabled) {
- return enabled == session->topPeers().disabled();
- }) | rpl::start_with_next([=](bool enabled) {
- session->topPeers().toggleDisabled(!enabled);
- }, container->lifetime());
- Ui::AddSkip(container);
- Ui::AddDividerText(container, tr::lng_settings_top_peers_about());
- }
- void SetupSelfDestruction(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger) {
- Ui::AddSkip(container);
- Ui::AddSubsectionTitle(container, tr::lng_settings_destroy_title());
- const auto session = &controller->session();
- std::move(
- updateTrigger
- ) | rpl::start_with_next([=] {
- session->api().selfDestruct().reload();
- }, container->lifetime());
- const auto label = [&] {
- return session->api().selfDestruct().daysAccountTTL(
- ) | rpl::map(SelfDestructionBox::DaysLabel);
- };
- AddButtonWithLabel(
- container,
- tr::lng_settings_destroy_if(),
- label(),
- st::settingsButtonNoIcon
- )->addClickHandler([=] {
- controller->show(Box<SelfDestructionBox>(
- session,
- SelfDestructionBox::Type::Account,
- session->api().selfDestruct().daysAccountTTL()));
- });
- Ui::AddSkip(container);
- }
- void ClearPaymentInfoBoxBuilder(
- not_null<Ui::GenericBox*> box,
- not_null<Main::Session*> session) {
- box->setTitle(tr::lng_clear_payment_info_title());
- const auto checkboxPadding = style::margins(
- st::boxRowPadding.left(),
- st::boxRowPadding.left(),
- st::boxRowPadding.right(),
- st::boxRowPadding.bottom());
- const auto label = box->addRow(object_ptr<Ui::FlatLabel>(
- box,
- tr::lng_clear_payment_info_sure(),
- st::boxLabel));
- const auto shipping = box->addRow(
- object_ptr<Ui::Checkbox>(
- box,
- tr::lng_clear_payment_info_shipping(tr::now),
- true,
- st::defaultBoxCheckbox),
- checkboxPadding);
- const auto payment = box->addRow(
- object_ptr<Ui::Checkbox>(
- box,
- tr::lng_clear_payment_info_payment(tr::now),
- true,
- st::defaultBoxCheckbox),
- checkboxPadding);
- using Flags = MTPpayments_ClearSavedInfo::Flags;
- const auto flags = box->lifetime().make_state<Flags>();
- box->addButton(tr::lng_clear_payment_info_clear(), [=] {
- using Flag = Flags::Enum;
- *flags = (shipping->checked() ? Flag::f_info : Flag(0))
- | (payment->checked() ? Flag::f_credentials : Flag(0));
- delete label;
- delete shipping;
- delete payment;
- box->addRow(object_ptr<Ui::FlatLabel>(
- box,
- tr::lng_clear_payment_info_confirm(),
- st::boxLabel));
- box->clearButtons();
- box->addButton(tr::lng_clear_payment_info_clear(), [=] {
- session->api().request(MTPpayments_ClearSavedInfo(
- MTP_flags(*flags)
- )).send();
- box->closeBox();
- }, st::attentionBoxButton);
- box->addButton(tr::lng_cancel(), [=] { box->closeBox(); });
- }, st::attentionBoxButton);
- box->addButton(tr::lng_cancel(), [=] { box->closeBox(); });
- }
- auto ClearPaymentInfoBox(not_null<Main::Session*> session) {
- return Box(ClearPaymentInfoBoxBuilder, session);
- }
- void SetupBotsAndWebsites(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container) {
- Ui::AddSkip(container);
- Ui::AddSubsectionTitle(container, tr::lng_settings_security_bots());
- const auto session = &controller->session();
- container->add(object_ptr<Button>(
- container,
- tr::lng_settings_clear_payment_info(),
- st::settingsButtonNoIcon
- ))->addClickHandler([=] {
- controller->show(ClearPaymentInfoBox(session));
- });
- Ui::AddSkip(container);
- Ui::AddDivider(container);
- }
- void SetupConfirmationExtensions(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container) {
- if (Core::App().settings().noWarningExtensions().empty()
- && Core::App().settings().ipRevealWarning()) {
- return;
- }
- Ui::AddSkip(container);
- Ui::AddSubsectionTitle(container, tr::lng_settings_file_confirmations());
- container->add(object_ptr<Button>(
- container,
- tr::lng_settings_edit_extensions(),
- st::settingsButtonNoIcon
- ))->addClickHandler([=] {
- controller->show(Box(OpenFileConfirmationsBox));
- });
- Ui::AddSkip(container);
- Ui::AddDividerText(container, tr::lng_settings_edit_extensions_about());
- }
- void SetupBlockedList(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger,
- Fn<void(Type)> showOther) {
- const auto session = &controller->session();
- auto blockedCount = rpl::combine(
- BlockedPeersCount(session),
- tr::lng_settings_no_blocked_users()
- ) | rpl::map([](int count, const QString &none) {
- return count ? QString::number(count) : none;
- });
- const auto blockedPeers = AddButtonWithLabel(
- container,
- tr::lng_settings_blocked_users(),
- std::move(blockedCount),
- st::settingsButton,
- { &st::menuIconBlock });
- blockedPeers->addClickHandler([=] {
- showOther(Blocked::Id());
- });
- std::move(
- updateTrigger
- ) | rpl::start_with_next([=] {
- session->api().blockedPeers().reload();
- }, blockedPeers->lifetime());
- }
- void SetupWebsitesList(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger,
- Fn<void(Type)> showOther) {
- std::move(
- updateTrigger
- ) | rpl::start_with_next([=] {
- controller->session().api().websites().reload();
- }, container->lifetime());
- auto count = controller->session().api().websites().totalValue();
- auto countText = rpl::duplicate(
- count
- ) | rpl::filter(rpl::mappers::_1 > 0) | rpl::map([](int count) {
- return QString::number(count);
- });
- const auto wrap = container->add(
- object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
- container,
- object_ptr<Ui::VerticalLayout>(container)));
- const auto inner = wrap->entity();
- AddButtonWithLabel(
- inner,
- tr::lng_settings_logged_in(),
- std::move(countText),
- st::settingsButton,
- { &st::menuIconIpAddress }
- )->addClickHandler([=] {
- showOther(Websites::Id());
- });
- wrap->toggleOn(std::move(count) | rpl::map(rpl::mappers::_1 > 0));
- wrap->finishAnimating();
- }
- void SetupSessionsList(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger,
- Fn<void(Type)> showOther) {
- std::move(
- updateTrigger
- ) | rpl::start_with_next([=] {
- controller->session().api().authorizations().reload();
- }, container->lifetime());
- auto count = controller->session().api().authorizations().totalValue(
- ) | rpl::map([](int count) {
- return count ? QString::number(count) : QString();
- });
- AddButtonWithLabel(
- container,
- tr::lng_settings_show_sessions(),
- std::move(count),
- st::settingsButton,
- { &st::menuIconDevices }
- )->addClickHandler([=] {
- showOther(Sessions::Id());
- });
- Ui::AddSkip(container);
- Ui::AddDividerText(container, tr::lng_settings_sessions_about());
- }
- void SetupGlobalTTLList(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger,
- Fn<void(Type)> showOther) {
- const auto session = &controller->session();
- auto ttlLabel = rpl::combine(
- session->api().selfDestruct().periodDefaultHistoryTTL(),
- tr::lng_settings_ttl_after_off()
- ) | rpl::map([](int ttl, const QString &none) {
- return ttl ? Ui::FormatTTL(ttl) : none;
- });
- const auto globalTTLButton = AddButtonWithLabel(
- container,
- tr::lng_settings_ttl_title(),
- std::move(ttlLabel),
- st::settingsButton,
- { &st::menuIconTTL });
- globalTTLButton->addClickHandler([=] {
- showOther(GlobalTTLId());
- });
- std::move(
- updateTrigger
- ) | rpl::start_with_next([=] {
- session->api().selfDestruct().reload();
- }, container->lifetime());
- }
- void SetupSecurity(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger,
- Fn<void(Type)> showOther) {
- Ui::AddSkip(container, st::settingsPrivacySkip);
- Ui::AddSubsectionTitle(container, tr::lng_settings_security());
- SetupCloudPassword(controller, container, showOther);
- SetupGlobalTTLList(
- controller,
- container,
- rpl::duplicate(updateTrigger),
- showOther);
- SetupLocalPasscode(controller, container, showOther);
- SetupBlockedList(
- controller,
- container,
- rpl::duplicate(updateTrigger),
- showOther);
- SetupWebsitesList(
- controller,
- container,
- rpl::duplicate(updateTrigger),
- showOther);
- SetupSessionsList(
- controller,
- container,
- rpl::duplicate(updateTrigger),
- showOther);
- }
- } // namespace
- void SetupSensitiveContent(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<> updateTrigger) {
- using namespace rpl::mappers;
- const auto wrap = container->add(
- object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
- container,
- object_ptr<Ui::VerticalLayout>(container)));
- const auto inner = wrap->entity();
- Ui::AddSkip(inner);
- Ui::AddSubsectionTitle(inner, tr::lng_settings_sensitive_title());
- const auto session = &controller->session();
- std::move(
- updateTrigger
- ) | rpl::start_with_next([=] {
- session->api().sensitiveContent().reload();
- }, container->lifetime());
- inner->add(object_ptr<Button>(
- inner,
- tr::lng_settings_sensitive_disable_filtering(),
- st::settingsButtonNoIcon
- ))->toggleOn(
- session->api().sensitiveContent().enabled()
- )->toggledChanges(
- ) | rpl::filter([=](bool toggled) {
- return toggled != session->api().sensitiveContent().enabledCurrent();
- }) | rpl::start_with_next([=](bool toggled) {
- session->api().sensitiveContent().update(toggled);
- }, container->lifetime());
- Ui::AddSkip(inner);
- Ui::AddDividerText(inner, tr::lng_settings_sensitive_about());
- wrap->toggleOn(session->api().sensitiveContent().canChange());
- }
- int ExceptionUsersCount(const std::vector<not_null<PeerData*>> &exceptions) {
- const auto add = [](int already, not_null<PeerData*> peer) {
- if (const auto chat = peer->asChat()) {
- return already + chat->count;
- } else if (const auto channel = peer->asChannel()) {
- return already + channel->membersCount();
- }
- return already + 1;
- };
- return ranges::accumulate(exceptions, 0, add);
- }
- bool CheckEditCloudPassword(not_null<::Main::Session*> session) {
- const auto current = session->api().cloudPassword().stateCurrent();
- Assert(current.has_value());
- return !current->outdatedClient;
- }
- object_ptr<Ui::BoxContent> EditCloudPasswordBox(not_null<Main::Session*> session) {
- const auto current = session->api().cloudPassword().stateCurrent();
- Assert(current.has_value());
- auto result = Box<PasscodeBox>(
- session,
- PasscodeBox::CloudFields::From(*current));
- const auto box = result.data();
- rpl::merge(
- box->newPasswordSet() | rpl::to_empty,
- box->passwordReloadNeeded()
- ) | rpl::start_with_next([=] {
- session->api().cloudPassword().reload();
- }, box->lifetime());
- box->clearUnconfirmedPassword(
- ) | rpl::start_with_next([=] {
- session->api().cloudPassword().clearUnconfirmedPassword();
- }, box->lifetime());
- return result;
- }
- void RemoveCloudPassword(not_null<Window::SessionController*> controller) {
- const auto session = &controller->session();
- const auto current = session->api().cloudPassword().stateCurrent();
- Assert(current.has_value());
- if (!current->hasPassword) {
- session->api().cloudPassword().clearUnconfirmedPassword();
- return;
- }
- auto fields = PasscodeBox::CloudFields::From(*current);
- fields.turningOff = true;
- auto box = Box<PasscodeBox>(session, fields);
- rpl::merge(
- box->newPasswordSet() | rpl::to_empty,
- box->passwordReloadNeeded()
- ) | rpl::start_with_next([=] {
- session->api().cloudPassword().reload();
- }, box->lifetime());
- box->clearUnconfirmedPassword(
- ) | rpl::start_with_next([=] {
- session->api().cloudPassword().clearUnconfirmedPassword();
- }, box->lifetime());
- controller->show(std::move(box));
- }
- object_ptr<Ui::BoxContent> CloudPasswordAppOutdatedBox() {
- const auto callback = [=](Fn<void()> &&close) {
- Core::UpdateApplication();
- close();
- };
- return Ui::MakeConfirmBox({
- .text = tr::lng_passport_app_out_of_date(),
- .confirmed = callback,
- .confirmText = tr::lng_menu_update(),
- });
- }
- not_null<Ui::SettingsButton*> AddPrivacyButton(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container,
- rpl::producer<QString> label,
- IconDescriptor &&descriptor,
- Privacy::Key key,
- Fn<std::unique_ptr<EditPrivacyController>()> controllerFactory,
- const style::SettingsButton *stOverride) {
- const auto shower = Ui::CreateChild<rpl::lifetime>(container.get());
- const auto session = &controller->session();
- const auto button = AddButtonWithLabel(
- container,
- std::move(label),
- PrivacyString(session, key),
- stOverride ? *stOverride : st::settingsButtonNoIcon,
- std::move(descriptor));
- button->addClickHandler([=] {
- *shower = session->api().userPrivacy().value(
- key
- ) | rpl::take(
- 1
- ) | rpl::start_with_next([=](const Privacy::Rule &value) {
- controller->show(Box<EditPrivacyBox>(
- controller,
- controllerFactory(),
- value));
- });
- });
- return button;
- }
- void SetupArchiveAndMute(
- not_null<Window::SessionController*> controller,
- not_null<Ui::VerticalLayout*> container) {
- using namespace rpl::mappers;
- const auto wrap = container->add(
- object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
- container,
- object_ptr<Ui::VerticalLayout>(container)));
- const auto inner = wrap->entity();
- Ui::AddSkip(inner);
- Ui::AddSubsectionTitle(inner, tr::lng_settings_new_unknown());
- const auto session = &controller->session();
- const auto privacy = &session->api().globalPrivacy();
- privacy->reload();
- inner->add(object_ptr<Button>(
- inner,
- tr::lng_settings_auto_archive(),
- st::settingsButtonNoIcon
- ))->toggleOn(
- privacy->archiveAndMute()
- )->toggledChanges(
- ) | rpl::filter([=](bool toggled) {
- return toggled != privacy->archiveAndMuteCurrent();
- }) | rpl::start_with_next([=](bool toggled) {
- privacy->updateArchiveAndMute(toggled);
- }, container->lifetime());
- Ui::AddSkip(inner);
- Ui::AddDividerText(inner, tr::lng_settings_auto_archive_about());
- auto shown = rpl::single(
- false
- ) | rpl::then(session->api().globalPrivacy().showArchiveAndMute(
- ) | rpl::filter(_1) | rpl::take(1));
- auto premium = Data::AmPremiumValue(&controller->session());
- using namespace rpl::mappers;
- wrap->toggleOn(rpl::combine(
- std::move(shown),
- std::move(premium),
- _1 || _2));
- }
- PrivacySecurity::PrivacySecurity(
- QWidget *parent,
- not_null<Window::SessionController*> controller)
- : Section(parent) {
- setupContent(controller);
- [[maybe_unused]] auto preload = base::SystemUnlockStatus();
- }
- rpl::producer<QString> PrivacySecurity::title() {
- return tr::lng_settings_section_privacy();
- }
- void PrivacySecurity::setupContent(
- not_null<Window::SessionController*> controller) {
- const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
- auto updateOnTick = rpl::single(
- ) | rpl::then(base::timer_each(kUpdateTimeout));
- const auto trigger = [=] {
- return rpl::duplicate(updateOnTick);
- };
- SetupSecurity(controller, content, trigger(), showOtherMethod());
- SetupPrivacy(controller, content, trigger());
- SetupTopPeers(controller, content);
- SetupArchiveAndMute(controller, content);
- SetupConfirmationExtensions(controller, content);
- SetupBotsAndWebsites(controller, content);
- SetupSelfDestruction(controller, content, trigger());
- Ui::ResizeFitChild(this, content);
- }
- } // namespace Settings
|