/* 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_blocked_peers.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 "settings/settings_common.h" #include "settings/settings_privacy_controllers.h" #include "base/timer_rpl.h" #include "base/unixtime.h" #include "boxes/peer_list_box.h" #include "boxes/edit_privacy_box.h" #include "boxes/passcode_box.h" #include "boxes/auto_lock_box.h" #include "boxes/sessions_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/wrap/vertical_layout.h" #include "ui/wrap/slide_wrap.h" #include "ui/wrap/fade_wrap.h" #include "ui/widgets/shadow.h" #include "ui/widgets/labels.h" #include "ui/widgets/buttons.h" #include "ui/widgets/checkbox.h" #include "ui/layers/generic_box.h" #include "calls/calls_instance.h" #include "core/core_cloud_password.h" #include "core/update_checker.h" #include "base/platform/base_platform_last_input.h" #include "lang/lang_keys.h" #include "data/data_session.h" #include "data/data_chat.h" #include "data/data_channel.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 "facades.h" #include "styles/style_settings.h" #include "styles/style_layers.h" #include "styles/style_boxes.h" #include namespace Settings { namespace { constexpr auto kUpdateTimeout = 60 * crl::time(1000); using Privacy = Api::UserPrivacy; QString PrivacyBase(Privacy::Key key, Privacy::Option option) { using Key = Privacy::Key; using Option = Privacy::Option; switch (key) { case Key::CallsPeer2Peer: switch (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); } Unexpected("Value in Privacy::Option."); default: switch (option) { case Option::Everyone: return tr::lng_edit_privacy_everyone(tr::now); case Option::Contacts: return tr::lng_edit_privacy_contacts(tr::now); case Option::Nobody: return tr::lng_edit_privacy_nobody(tr::now); } Unexpected("Value in Privacy::Option."); } } rpl::producer 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)) { add.push_back("-" + QString::number(never)); } if (const auto always = ExceptionUsersCount(value.always)) { add.push_back("+" + QString::number(always)); } if (!add.isEmpty()) { return PrivacyBase(key, value.option) + " (" + add.join(", ") + ")"; } else { return PrivacyBase(key, value.option); } }); } rpl::producer 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 controller, not_null container, rpl::producer<> updateTrigger) { AddSkip(container, st::settingsPrivacySkip); AddSubsectionTitle(container, tr::lng_settings_privacy_title()); const auto session = &controller->session(); using Key = Privacy::Key; const auto add = [&]( rpl::producer label, IconDescriptor &&descriptor, Key key, auto controllerFactory) { AddPrivacyButton( controller, container, std::move(label), std::move(descriptor), key, controllerFactory); }; add( tr::lng_settings_phone_number_privacy(), { &st::settingsIconCalls, kIconGreen }, Key::PhoneNumber, [=] { return std::make_unique( controller); }); add( tr::lng_settings_last_seen(), { &st::settingsIconOnline, kIconLightBlue }, Key::LastSeen, [=] { return std::make_unique(session); }); add( tr::lng_settings_forwards_privacy(), { &st::settingsIconForward, kIconLightOrange }, Key::Forwards, [=] { return std::make_unique( controller); }); add( tr::lng_settings_profile_photo_privacy(), { &st::settingsIconAccount, kIconRed }, Key::ProfilePhoto, [] { return std::make_unique(); }); add( tr::lng_settings_calls(), { &st::settingsIconVideoCalls, kIconGreen }, Key::Calls, [] { return std::make_unique(); }); add( tr::lng_settings_groups_invite(), { &st::settingsIconGroup, kIconDarkBlue }, Key::Invites, [] { return std::make_unique(); }); session->api().userPrivacy().reload(Api::UserPrivacy::Key::AddedByPhone); AddSkip(container, st::settingsPrivacySecurityPadding); AddDividerText(container, tr::lng_settings_group_privacy_about()); } void SetupArchiveAndMute( not_null controller, not_null container) { using namespace rpl::mappers; const auto wrap = container->add( object_ptr>( container, object_ptr(container))); const auto inner = wrap->entity(); AddSkip(inner); AddSubsectionTitle(inner, tr::lng_settings_new_unknown()); const auto session = &controller->session(); const auto privacy = &session->api().globalPrivacy(); privacy->reload(); AddButton( 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->update(toggled); }, container->lifetime()); AddSkip(inner); AddDividerText(inner, tr::lng_settings_auto_archive_about()); using namespace rpl::mappers; wrap->toggleOn(rpl::single( false ) | rpl::then( session->api().globalPrivacy().showArchiveAndMute( ) | rpl::filter(_1) | rpl::take(1) )); } void SetupLocalPasscode( not_null controller, not_null container) { AddSkip(container); AddDivider(container); AddSkip(container); AddSubsectionTitle(container, tr::lng_settings_passcode_title()); auto has = rpl::single(rpl::empty) | rpl::then( controller->session().domain().local().localPasscodeChanged() ) | rpl::map([=] { return controller->session().domain().local().hasLocalPasscode(); }); auto text = rpl::combine( tr::lng_passcode_change(), tr::lng_passcode_turn_on(), base::duplicate(has), [](const QString &change, const QString &create, bool has) { return has ? change : create; }); AddButton( container, std::move(text), st::settingsButton, { &st::settingsIconLock, kIconGreen } )->addClickHandler([=] { controller->show(Box(&controller->session(), false)); }); const auto wrap = container->add( object_ptr>( container, object_ptr(container))); const auto inner = wrap->entity(); AddButton( inner, tr::lng_settings_passcode_disable(), st::settingsButton, { &st::settingsIconMinus, kIconRed } )->addClickHandler([=] { controller->show(Box(&controller->session(), true)); }); const auto autoLockBoxClosing = container->lifetime().make_state>(); const auto label = base::Platform::LastUserInputTimeSupported() ? tr::lng_passcode_autolock_away : tr::lng_passcode_autolock_inactive; auto value = autoLockBoxClosing->events_starting_with( {} ) | rpl::map([] { const auto autolock = Core::App().settings().autoLock(); const auto hours = autolock / 3600; const auto minutes = (autolock - (hours * 3600)) / 60; return (hours && minutes) ? tr::lng_passcode_autolock_hours_minutes( tr::now, lt_hours_count, QString::number(hours), lt_minutes_count, QString::number(minutes)) : minutes ? tr::lng_minutes(tr::now, lt_count, minutes) : tr::lng_hours(tr::now, lt_count, hours); }); AddButtonWithLabel( inner, label(), std::move(value), st::settingsButton, { &st::settingsIconTimer, kIconGreen } )->addClickHandler([=] { const auto box = controller->show(Box()); box->boxClosing( ) | rpl::start_to_stream(*autoLockBoxClosing, box->lifetime()); }); wrap->toggleOn(base::duplicate(has)); } void SetupCloudPassword( not_null controller, not_null container) { using namespace rpl::mappers; using State = Core::CloudPasswordState; AddSkip(container); AddDivider(container); AddSkip(container); AddSubsectionTitle(container, tr::lng_settings_password_title()); const auto session = &controller->session(); auto has = rpl::single( false ) | rpl::then(controller->session().api().cloudPassword().state( ) | rpl::map([](const State &state) { return state.request || state.unknownAlgorithm || !state.unconfirmedPattern.isEmpty(); })) | rpl::distinct_until_changed(); auto pattern = session->api().cloudPassword().state( ) | rpl::map([](const State &state) { return state.unconfirmedPattern; }); auto confirmation = rpl::single( tr::lng_profile_loading(tr::now) ) | rpl::then(rpl::duplicate( pattern ) | rpl::filter([](const QString &pattern) { return !pattern.isEmpty(); }) | rpl::map([](const QString &pattern) { return tr::lng_cloud_password_waiting_code(tr::now, lt_email, pattern); })); auto unconfirmed = rpl::duplicate( pattern ) | rpl::map([](const QString &pattern) { return !pattern.isEmpty(); }); auto noconfirmed = rpl::single( true ) | rpl::then(rpl::duplicate( unconfirmed )); auto resetAt = session->api().cloudPassword().state( ) | rpl::map([](const State &state) { return state.pendingResetDate; }); const auto label = container->add( object_ptr>( container, object_ptr( container, base::duplicate(confirmation), st::settingsCloudPasswordLabel), QMargins( st::settingsButtonNoIcon.padding.left(), st::settingsButtonNoIcon.padding.top(), st::settingsButtonNoIcon.padding.right(), (st::settingsButtonNoIcon.height - st::settingsCloudPasswordLabel.style.font->height + st::settingsButtonNoIcon.padding.bottom())))); label->toggleOn(base::duplicate(noconfirmed))->setDuration(0); std::move( confirmation ) | rpl::start_with_next([=] { container->resizeToWidth(container->width()); }, label->lifetime()); auto text = rpl::combine( tr::lng_cloud_password_set(), tr::lng_cloud_password_edit(), base::duplicate(has) ) | rpl::map([](const QString &set, const QString &edit, bool has) { return has ? edit : set; }); const auto change = container->add( object_ptr>( container, CreateButton( container, std::move(text), st::settingsButton, { &st::settingsIconKey, kIconLightBlue }))); change->toggleOn(rpl::duplicate( noconfirmed ) | rpl::map( !_1 ))->setDuration(0); change->entity()->addClickHandler([=] { if (CheckEditCloudPassword(session)) { controller->show(EditCloudPasswordBox(session)); } else { controller->show(CloudPasswordAppOutdatedBox()); } }); const auto confirm = container->add( object_ptr>( container, CreateButton( container, tr::lng_cloud_password_confirm(), st::settingsButton, { &st::settingsIconEmail, kIconLightOrange }))); confirm->toggleOn(rpl::single( false ) | rpl::then(rpl::duplicate( unconfirmed )))->setDuration(0); confirm->entity()->addClickHandler([=] { const auto state = session->api().cloudPassword().stateCurrent(); if (!state) { return; } auto validation = ConfirmRecoveryEmail( &controller->session(), state->unconfirmedPattern); std::move( validation.reloadRequests ) | rpl::start_with_next([=] { session->api().cloudPassword().reload(); }, validation.box->lifetime()); std::move( validation.cancelRequests ) | rpl::start_with_next([=] { session->api().cloudPassword().clearUnconfirmedPassword(); }, validation.box->lifetime()); controller->show(std::move(validation.box)); }); const auto remove = [=] { if (CheckEditCloudPassword(session)) { RemoveCloudPassword(controller); } else { controller->show(CloudPasswordAppOutdatedBox()); } }; const auto disable = container->add( object_ptr>( container, CreateButton( container, tr::lng_settings_password_disable(), st::settingsButton, { &st::settingsIconMinus, kIconRed }))); disable->toggleOn(rpl::combine( rpl::duplicate(has), rpl::duplicate(noconfirmed), _1 && !_2)); disable->entity()->addClickHandler(remove); auto resetInSeconds = rpl::duplicate( resetAt ) | rpl::filter([](TimeId time) { return time != 0; }) | rpl::map([](TimeId time) { return rpl::single(rpl::empty) | rpl::then(base::timer_each( 999 )) | rpl::map([=] { const auto now = base::unixtime::now(); return (time - now); }) | rpl::distinct_until_changed( ) | rpl::take_while([](TimeId left) { return left > 0; }) | rpl::then(rpl::single(TimeId(0))); }) | rpl::flatten_latest( ) | rpl::start_spawning(container->lifetime()); auto resetText = rpl::duplicate( resetInSeconds ) | rpl::map([](TimeId left) { return (left > 0); }) | rpl::distinct_until_changed( ) | rpl::map([](bool waiting) { return waiting ? tr::lng_cloud_password_reset_in() : tr::lng_cloud_password_reset_ready(); }) | rpl::flatten_latest(); constexpr auto kMinute = 60; constexpr auto kHour = 3600; constexpr auto kDay = 86400; auto resetLabel = rpl::duplicate( resetInSeconds ) | rpl::map([](TimeId left) { return (left >= kDay) ? ((left / kDay) * kDay) : (left >= kHour) ? ((left / kHour) * kHour) : (left >= kMinute) ? ((left / kMinute) * kMinute) : left; }) | rpl::distinct_until_changed( ) | rpl::map([](TimeId left) { const auto days = left / kDay; const auto hours = left / kHour; const auto minutes = left / kMinute; return days ? tr::lng_days(tr::now, lt_count, days) : hours ? tr::lng_hours(tr::now, lt_count, hours) : minutes ? tr::lng_minutes(tr::now, lt_count, minutes) : left ? tr::lng_seconds(tr::now, lt_count, left) : QString(); }); const auto reset = container->add( object_ptr>( container, CreateButton( container, rpl::duplicate(resetText), st::settingsButton)) )->setDuration(0); CreateRightLabel( reset->entity(), std::move(resetLabel), st::settingsButton, std::move(resetText)); reset->toggleOn(rpl::duplicate( resetAt ) | rpl::map([](TimeId time) { return time != 0; })); const auto sent = std::make_shared(false); reset->entity()->addClickHandler([=] { const auto api = &session->api(); const auto state = api->cloudPassword().stateCurrent(); const auto date = state ? state->pendingResetDate : TimeId(0); if (!date || *sent) { return; } else if (base::unixtime::now() >= date) { *sent = true; api->cloudPassword().resetPassword( ) | rpl::start_with_error_done([=](const QString &error) { *sent = false; }, [=] { *sent = false; }, container->lifetime()); } else { const auto cancel = [=] { Ui::hideLayer(); *sent = true; api->cloudPassword().cancelResetPassword( ) | rpl::start_with_error_done([=](const QString &error) { *sent = false; }, [=] { *sent = false; }, container->lifetime()); }; Ui::show(Ui::MakeConfirmBox({ .text = tr::lng_cloud_password_reset_cancel_sure(), .confirmed = cancel, .confirmText = tr::lng_box_yes(), .cancelText = tr::lng_box_no(), })); } }); const auto abort = container->add( object_ptr>( container, CreateButton( container, tr::lng_settings_password_abort(), st::settingsAttentionButton))); abort->toggleOn(rpl::combine( rpl::duplicate(has), rpl::duplicate(noconfirmed), _1 && _2)); abort->entity()->addClickHandler(remove); const auto reloadOnActivation = [=](Qt::ApplicationState state) { if (label->toggled() && state == Qt::ApplicationActive) { controller->session().api().cloudPassword().reload(); } }; QObject::connect( static_cast(QCoreApplication::instance()), &QGuiApplication::applicationStateChanged, label, reloadOnActivation); session->api().cloudPassword().reload(); AddSkip(container); AddDivider(container); } void SetupSensitiveContent( not_null controller, not_null container, rpl::producer<> updateTrigger) { using namespace rpl::mappers; const auto wrap = container->add( object_ptr>( container, object_ptr(container))); const auto inner = wrap->entity(); AddSkip(inner); 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()); AddButton( 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()); AddSkip(inner); AddDividerText(inner, tr::lng_settings_sensitive_about()); wrap->toggleOn(session->api().sensitiveContent().canChange()); } void SetupSelfDestruction( not_null controller, not_null container, rpl::producer<> updateTrigger) { AddSkip(container); 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().days( ) | rpl::map( SelfDestructionBox::DaysLabel ); }; AddButtonWithLabel( container, tr::lng_settings_destroy_if(), label(), st::settingsButtonNoIcon )->addClickHandler([=] { controller->show(Box( session, SelfDestructionBox::Type::Account, session->api().selfDestruct().days())); }); AddSkip(container); } void ClearPaymentInfoBoxBuilder( not_null box, not_null 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( box, tr::lng_clear_payment_info_sure(), st::boxLabel)); const auto shipping = box->addRow( object_ptr( box, tr::lng_clear_payment_info_shipping(tr::now), true, st::defaultBoxCheckbox), checkboxPadding); const auto payment = box->addRow( object_ptr( 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(); 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( 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 session) { return Box(ClearPaymentInfoBoxBuilder, session); } void SetupBotsAndWebsites( not_null controller, not_null container) { AddSkip(container); AddSubsectionTitle(container, tr::lng_settings_security_bots()); const auto session = &controller->session(); AddButton( container, tr::lng_settings_clear_payment_info(), st::settingsButtonNoIcon )->addClickHandler([=] { controller->show(ClearPaymentInfoBox(session)); }); AddSkip(container); } void SetupBlockedList( not_null controller, not_null container, rpl::producer<> updateTrigger, Fn 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::settingsIconMinus, kIconRed }); blockedPeers->addClickHandler([=] { const auto initBox = [=](not_null box) { box->addButton(tr::lng_close(), [=] { box->closeBox(); }); box->addLeftButton(tr::lng_blocked_list_add(), [=] { BlockedBoxController::BlockNewPeer(controller); }); }; controller->show(Box( std::make_unique(controller), initBox)); }); std::move( updateTrigger ) | rpl::start_with_next([=] { session->api().blockedPeers().reload(); }, blockedPeers->lifetime()); } void SetupSessionsList( not_null controller, not_null container, rpl::producer<> updateTrigger, Fn showOther) { std::move( updateTrigger ) | rpl::start_with_next([=] { controller->session().api().authorizations().reload(); }, container->lifetime()); auto count = controller->session().api().authorizations().totalChanges( ) | rpl::map([](int count) { return count ? QString::number(count) : QString(); }); AddButtonWithLabel( container, tr::lng_settings_show_sessions(), std::move(count), st::settingsButton, { &st::settingsIconLaptop, kIconLightOrange } )->addClickHandler([=] { showOther(Sessions::Id()); }); } void SetupSecurity( not_null controller, not_null container, rpl::producer<> updateTrigger, Fn showOther) { AddSkip(container, st::settingsPrivacySkip); AddSubsectionTitle(container, tr::lng_settings_security()); SetupBlockedList( controller, container, rpl::duplicate(updateTrigger), showOther); SetupSessionsList( controller, container, rpl::duplicate(updateTrigger), showOther); SetupLocalPasscode(controller, container); SetupCloudPassword(controller, container); } } // namespace int ExceptionUsersCount(const std::vector> &exceptions) { const auto add = [](int already, not_null 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()); if (!current->unknownAlgorithm && !v::is_null(current->newPassword) && !v::is_null(current->newSecureSecret)) { return true; } return false; } object_ptr EditCloudPasswordBox(not_null session) { const auto current = session->api().cloudPassword().stateCurrent(); Assert(current.has_value()); auto result = Box( 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 controller) { const auto session = &controller->session(); const auto current = session->api().cloudPassword().stateCurrent(); Assert(current.has_value()); if (!current->request) { session->api().cloudPassword().clearUnconfirmedPassword(); return; } auto fields = PasscodeBox::CloudFields::From(*current); fields.turningOff = true; auto box = Box(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 CloudPasswordAppOutdatedBox() { const auto callback = [=](Fn &&close) { Core::UpdateApplication(); close(); }; return Ui::MakeConfirmBox({ .text = tr::lng_passport_app_out_of_date(), .confirmed = callback, .confirmText = tr::lng_menu_update(), }); } void AddPrivacyButton( not_null controller, not_null container, rpl::producer label, IconDescriptor &&descriptor, Privacy::Key key, Fn()> controllerFactory) { const auto shower = Ui::CreateChild(container.get()); const auto session = &controller->session(); AddButtonWithLabel( container, std::move(label), PrivacyString(session, key), st::settingsButton, std::move(descriptor) )->addClickHandler([=] { *shower = session->api().userPrivacy().value( key ) | rpl::take( 1 ) | rpl::start_with_next([=](const Privacy::Rule &value) { controller->show( Box(controller, controllerFactory(), value), Ui::LayerOption::KeepOther); }); }); } PrivacySecurity::PrivacySecurity( QWidget *parent, not_null controller) : Section(parent) { setupContent(controller); } rpl::producer PrivacySecurity::title() { return tr::lng_settings_section_privacy(); } rpl::producer PrivacySecurity::sectionShowOther() { return _showOther.events(); } void PrivacySecurity::setupContent( not_null controller) { const auto content = Ui::CreateChild(this); auto updateOnTick = rpl::single( ) | rpl::then(base::timer_each(kUpdateTimeout)); const auto trigger = [=] { return rpl::duplicate(updateOnTick); }; SetupPrivacy(controller, content, trigger()); SetupSecurity(controller, content, trigger(), [=](Type type) { _showOther.fire_copy(type); }); #if !defined OS_MAC_STORE && !defined OS_WIN_STORE SetupSensitiveContent(controller, content, trigger()); #else // !OS_MAC_STORE && !OS_WIN_STORE AddDivider(content); #endif // !OS_MAC_STORE && !OS_WIN_STORE SetupArchiveAndMute(controller, content); SetupBotsAndWebsites(controller, content); AddDivider(content); SetupSelfDestruction(controller, content, trigger()); Ui::ResizeFitChild(this, content); } } // namespace Settings