583 lines
15 KiB
C++
583 lines
15 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "settings/settings_privacy_security.h"
|
|
|
|
#include "settings/settings_common.h"
|
|
#include "settings/settings_privacy_controllers.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 "boxes/confirm_box.h"
|
|
#include "boxes/self_destruction_box.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 "calls/calls_instance.h"
|
|
#include "core/core_cloud_password.h"
|
|
#include "core/update_checker.h"
|
|
#include "info/profile/info_profile_button.h"
|
|
#include "platform/platform_specific.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_chat.h"
|
|
#include "data/data_channel.h"
|
|
#include "auth_session.h"
|
|
#include "apiwrap.h"
|
|
#include "styles/style_settings.h"
|
|
#include "styles/style_boxes.h"
|
|
|
|
namespace Settings {
|
|
namespace {
|
|
|
|
using Privacy = ApiWrap::Privacy;
|
|
|
|
rpl::producer<> PasscodeChanges() {
|
|
return rpl::single(
|
|
rpl::empty_value()
|
|
) | rpl::then(base::ObservableViewer(
|
|
Global::RefLocalPasscodeChanged()
|
|
));
|
|
}
|
|
|
|
QString PrivacyBase(Privacy::Key key, Privacy::Option option) {
|
|
const auto phrase = [&] {
|
|
using Key = Privacy::Key;
|
|
using Option = Privacy::Option;
|
|
switch (key) {
|
|
case Key::CallsPeer2Peer:
|
|
switch (option) {
|
|
case Option::Everyone:
|
|
return lng_edit_privacy_calls_p2p_everyone;
|
|
case Option::Contacts:
|
|
return lng_edit_privacy_calls_p2p_contacts;
|
|
case Option::Nobody:
|
|
return lng_edit_privacy_calls_p2p_nobody;
|
|
}
|
|
Unexpected("Value in Privacy::Option.");
|
|
default:
|
|
switch (option) {
|
|
case Option::Everyone: return lng_edit_privacy_everyone;
|
|
case Option::Contacts: return lng_edit_privacy_contacts;
|
|
case Option::Nobody: return lng_edit_privacy_nobody;
|
|
}
|
|
Unexpected("Value in Privacy::Option.");
|
|
}
|
|
}();
|
|
return lang(phrase);
|
|
}
|
|
|
|
rpl::producer<QString> PrivacyString(Privacy::Key key) {
|
|
Auth().api().reloadPrivacy(key);
|
|
return Auth().api().privacyValue(
|
|
key
|
|
) | rpl::map([=](const Privacy &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<int> BlockedUsersCount() {
|
|
Auth().api().reloadBlockedUsers();
|
|
return Auth().api().blockedUsersSlice(
|
|
) | rpl::map([=](const ApiWrap::BlockedUsersSlice &data) {
|
|
return data.total;
|
|
});
|
|
}
|
|
|
|
void SetupPrivacy(not_null<Ui::VerticalLayout*> container) {
|
|
AddSkip(container, st::settingsPrivacySkip);
|
|
AddSubsectionTitle(container, lng_settings_privacy_title);
|
|
|
|
auto count = BlockedUsersCount(
|
|
) | rpl::map([](int count) {
|
|
return count ? QString::number(count) : QString();
|
|
});
|
|
AddButtonWithLabel(
|
|
container,
|
|
lng_settings_blocked_users,
|
|
std::move(count),
|
|
st::settingsButton
|
|
)->addClickHandler([] {
|
|
const auto initBox = [](not_null<PeerListBox*> box) {
|
|
box->addButton(langFactory(lng_close), [=] {
|
|
box->closeBox();
|
|
});
|
|
box->addLeftButton(langFactory(lng_blocked_list_add), [] {
|
|
BlockedBoxController::BlockNewUser();
|
|
});
|
|
};
|
|
Ui::show(Box<PeerListBox>(
|
|
std::make_unique<BlockedBoxController>(),
|
|
initBox));
|
|
});
|
|
|
|
using Key = Privacy::Key;
|
|
const auto add = [&](LangKey label, Key key, auto controller) {
|
|
AddPrivacyButton(container, label, key, controller);
|
|
};
|
|
add(
|
|
lng_settings_phone_number_privacy,
|
|
Key::PhoneNumber,
|
|
[] { return std::make_unique<PhoneNumberPrivacyController>(); });
|
|
add(
|
|
lng_settings_last_seen,
|
|
Key::LastSeen,
|
|
[] { return std::make_unique<LastSeenPrivacyController>(); });
|
|
add(
|
|
lng_settings_forwards_privacy,
|
|
Key::Forwards,
|
|
[] { return std::make_unique<ForwardsPrivacyController>(); });
|
|
add(
|
|
lng_settings_profile_photo_privacy,
|
|
Key::ProfilePhoto,
|
|
[] { return std::make_unique<ProfilePhotoPrivacyController>(); });
|
|
add(
|
|
lng_settings_calls,
|
|
Key::Calls,
|
|
[] { return std::make_unique<CallsPrivacyController>(); });
|
|
add(
|
|
lng_settings_groups_invite,
|
|
Key::Invites,
|
|
[] { return std::make_unique<GroupsInvitePrivacyController>(); });
|
|
|
|
AddSkip(container, st::settingsPrivacySecurityPadding);
|
|
AddDividerText(
|
|
container,
|
|
Lang::Viewer(lng_settings_group_privacy_about));
|
|
}
|
|
|
|
not_null<Ui::SlideWrap<Ui::PlainShadow>*> AddSeparator(
|
|
not_null<Ui::VerticalLayout*> container) {
|
|
return container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::PlainShadow>>(
|
|
container,
|
|
object_ptr<Ui::PlainShadow>(container),
|
|
st::settingsSeparatorPadding));
|
|
}
|
|
|
|
void SetupLocalPasscode(not_null<Ui::VerticalLayout*> container) {
|
|
AddSkip(container);
|
|
AddSubsectionTitle(container, lng_settings_passcode_title);
|
|
|
|
auto has = PasscodeChanges(
|
|
) | rpl::map([] {
|
|
return Global::LocalPasscode();
|
|
});
|
|
auto text = rpl::combine(
|
|
Lang::Viewer(lng_passcode_change),
|
|
Lang::Viewer(lng_passcode_turn_on),
|
|
base::duplicate(has),
|
|
[](const QString &change, const QString &create, bool has) {
|
|
return has ? change : create;
|
|
});
|
|
container->add(
|
|
object_ptr<Button>(
|
|
container,
|
|
std::move(text),
|
|
st::settingsButton)
|
|
)->addClickHandler([] {
|
|
Ui::show(Box<PasscodeBox>(false));
|
|
});
|
|
|
|
const auto wrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(container)));
|
|
const auto inner = wrap->entity();
|
|
inner->add(
|
|
object_ptr<Button>(
|
|
inner,
|
|
Lang::Viewer(lng_settings_passcode_disable),
|
|
st::settingsButton)
|
|
)->addClickHandler([] {
|
|
Ui::show(Box<PasscodeBox>(true));
|
|
});
|
|
|
|
const auto label = Platform::LastUserInputTimeSupported()
|
|
? lng_passcode_autolock_away
|
|
: lng_passcode_autolock_inactive;
|
|
auto value = PasscodeChanges(
|
|
) | rpl::map([] {
|
|
const auto autolock = Global::AutoLock();
|
|
return (autolock % 3600)
|
|
? lng_passcode_autolock_minutes(lt_count, autolock / 60)
|
|
: lng_passcode_autolock_hours(lt_count, autolock / 3600);
|
|
});
|
|
|
|
AddButtonWithLabel(
|
|
inner,
|
|
label,
|
|
std::move(value),
|
|
st::settingsButton
|
|
)->addClickHandler([] {
|
|
Ui::show(Box<AutoLockBox>());
|
|
});
|
|
|
|
wrap->toggleOn(base::duplicate(has));
|
|
|
|
AddSkip(container);
|
|
}
|
|
|
|
bool CheckEditCloudPassword() {
|
|
const auto current = Auth().api().passwordStateCurrent();
|
|
Assert(current.has_value());
|
|
if (!current->unknownAlgorithm
|
|
&& current->newPassword
|
|
&& current->newSecureSecret) {
|
|
return true;
|
|
}
|
|
auto box = std::make_shared<QPointer<BoxContent>>();
|
|
const auto callback = [=] {
|
|
Core::UpdateApplication();
|
|
if (*box) (*box)->closeBox();
|
|
};
|
|
*box = Ui::show(Box<ConfirmBox>(
|
|
lang(lng_passport_app_out_of_date),
|
|
lang(lng_menu_update),
|
|
callback));
|
|
return false;
|
|
}
|
|
|
|
void EditCloudPassword() {
|
|
const auto current = Auth().api().passwordStateCurrent();
|
|
Assert(current.has_value());
|
|
|
|
const auto box = Ui::show(Box<PasscodeBox>(
|
|
current->request,
|
|
current->newPassword,
|
|
current->hasRecovery,
|
|
current->notEmptyPassport,
|
|
current->hint,
|
|
current->newSecureSecret));
|
|
|
|
rpl::merge(
|
|
box->newPasswordSet() | rpl::map([] { return rpl::empty_value(); }),
|
|
box->passwordReloadNeeded()
|
|
) | rpl::start_with_next([=] {
|
|
Auth().api().reloadPasswordState();
|
|
}, box->lifetime());
|
|
|
|
box->clearUnconfirmedPassword(
|
|
) | rpl::start_with_next([=] {
|
|
Auth().api().clearUnconfirmedPassword();
|
|
}, box->lifetime());
|
|
}
|
|
|
|
void RemoveCloudPassword() {
|
|
const auto current = Auth().api().passwordStateCurrent();
|
|
Assert(current.has_value());
|
|
|
|
if (!current->request) {
|
|
Auth().api().clearUnconfirmedPassword();
|
|
return;
|
|
}
|
|
const auto box = Ui::show(Box<PasscodeBox>(
|
|
current->request,
|
|
current->newPassword,
|
|
current->hasRecovery,
|
|
current->notEmptyPassport,
|
|
current->hint,
|
|
current->newSecureSecret,
|
|
true));
|
|
|
|
rpl::merge(
|
|
box->newPasswordSet(
|
|
) | rpl::map([] { return rpl::empty_value(); }),
|
|
box->passwordReloadNeeded()
|
|
) | rpl::start_with_next([=] {
|
|
Auth().api().reloadPasswordState();
|
|
}, box->lifetime());
|
|
|
|
box->clearUnconfirmedPassword(
|
|
) | rpl::start_with_next([=] {
|
|
Auth().api().clearUnconfirmedPassword();
|
|
}, box->lifetime());
|
|
}
|
|
|
|
void SetupCloudPassword(not_null<Ui::VerticalLayout*> container) {
|
|
using namespace rpl::mappers;
|
|
using State = Core::CloudPasswordState;
|
|
|
|
AddDivider(container);
|
|
AddSkip(container);
|
|
AddSubsectionTitle(container, lng_settings_password_title);
|
|
|
|
auto has = rpl::single(
|
|
false
|
|
) | rpl::then(Auth().api().passwordState(
|
|
) | rpl::map([](const State &state) {
|
|
return state.request
|
|
|| state.unknownAlgorithm
|
|
|| !state.unconfirmedPattern.isEmpty();
|
|
})) | rpl::distinct_until_changed();
|
|
auto pattern = Auth().api().passwordState(
|
|
) | rpl::map([](const State &state) {
|
|
return state.unconfirmedPattern;
|
|
});
|
|
auto confirmation = rpl::single(
|
|
lang(lng_profile_loading)
|
|
) | rpl::then(rpl::duplicate(
|
|
pattern
|
|
) | rpl::filter([](const QString &pattern) {
|
|
return !pattern.isEmpty();
|
|
}) | rpl::map([](const QString &pattern) {
|
|
return lng_cloud_password_waiting_code(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
|
|
));
|
|
const auto label = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::FlatLabel>>(
|
|
container,
|
|
object_ptr<Ui::FlatLabel>(
|
|
container,
|
|
base::duplicate(confirmation),
|
|
st::settingsCloudPasswordLabel),
|
|
QMargins(
|
|
st::settingsButton.padding.left(),
|
|
st::settingsButton.padding.top(),
|
|
st::settingsButton.padding.right(),
|
|
(st::settingsButton.height
|
|
- st::settingsCloudPasswordLabel.style.font->height
|
|
+ st::settingsButton.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(
|
|
Lang::Viewer(lng_cloud_password_set),
|
|
Lang::Viewer(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<Ui::SlideWrap<Button>>(
|
|
container,
|
|
object_ptr<Button>(
|
|
container,
|
|
std::move(text),
|
|
st::settingsButton)));
|
|
change->toggleOn(rpl::duplicate(
|
|
noconfirmed
|
|
) | rpl::map(
|
|
!_1
|
|
))->setDuration(0);
|
|
change->entity()->addClickHandler([] {
|
|
if (CheckEditCloudPassword()) {
|
|
EditCloudPassword();
|
|
}
|
|
});
|
|
|
|
const auto confirm = container->add(
|
|
object_ptr<Ui::SlideWrap<Button>>(
|
|
container,
|
|
object_ptr<Button>(
|
|
container,
|
|
Lang::Viewer(lng_cloud_password_confirm),
|
|
st::settingsButton)));
|
|
confirm->toggleOn(rpl::single(
|
|
false
|
|
) | rpl::then(rpl::duplicate(
|
|
unconfirmed
|
|
)))->setDuration(0);
|
|
confirm->entity()->addClickHandler([] {
|
|
const auto state = Auth().api().passwordStateCurrent();
|
|
if (!state) {
|
|
return;
|
|
}
|
|
auto validation = ConfirmRecoveryEmail(state->unconfirmedPattern);
|
|
|
|
std::move(
|
|
validation.reloadRequests
|
|
) | rpl::start_with_next([] {
|
|
Auth().api().reloadPasswordState();
|
|
}, validation.box->lifetime());
|
|
|
|
std::move(
|
|
validation.cancelRequests
|
|
) | rpl::start_with_next([] {
|
|
Auth().api().clearUnconfirmedPassword();
|
|
}, validation.box->lifetime());
|
|
|
|
Ui::show(std::move(validation.box));
|
|
});
|
|
|
|
const auto remove = [] {
|
|
if (CheckEditCloudPassword()) {
|
|
RemoveCloudPassword();
|
|
}
|
|
};
|
|
const auto disable = container->add(
|
|
object_ptr<Ui::SlideWrap<Button>>(
|
|
container,
|
|
object_ptr<Button>(
|
|
container,
|
|
Lang::Viewer(lng_settings_password_disable),
|
|
st::settingsButton)));
|
|
disable->toggleOn(rpl::combine(
|
|
rpl::duplicate(has),
|
|
rpl::duplicate(noconfirmed),
|
|
_1 && !_2));
|
|
disable->entity()->addClickHandler(remove);
|
|
|
|
const auto abort = container->add(
|
|
object_ptr<Ui::SlideWrap<Button>>(
|
|
container,
|
|
object_ptr<Button>(
|
|
container,
|
|
Lang::Viewer(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) {
|
|
Auth().api().reloadPasswordState();
|
|
}
|
|
};
|
|
QObject::connect(
|
|
static_cast<QGuiApplication*>(QCoreApplication::instance()),
|
|
&QGuiApplication::applicationStateChanged,
|
|
label,
|
|
reloadOnActivation);
|
|
|
|
Auth().api().reloadPasswordState();
|
|
|
|
AddSkip(container);
|
|
}
|
|
|
|
void SetupSelfDestruction(not_null<Ui::VerticalLayout*> container) {
|
|
AddDivider(container);
|
|
AddSkip(container);
|
|
AddSubsectionTitle(container, lng_settings_destroy_title);
|
|
|
|
Auth().api().reloadSelfDestruct();
|
|
const auto label = [] {
|
|
return Auth().api().selfDestructValue(
|
|
) | rpl::map(
|
|
SelfDestructionBox::DaysLabel
|
|
);
|
|
};
|
|
|
|
AddButtonWithLabel(
|
|
container,
|
|
lng_settings_destroy_if,
|
|
label(),
|
|
st::settingsButton
|
|
)->addClickHandler([] {
|
|
Ui::show(Box<SelfDestructionBox>(Auth().api().selfDestructValue()));
|
|
});
|
|
|
|
AddSkip(container);
|
|
}
|
|
|
|
void SetupSessionsList(not_null<Ui::VerticalLayout*> container) {
|
|
AddSkip(container);
|
|
AddSubsectionTitle(container, lng_settings_sessions_title);
|
|
|
|
AddButton(
|
|
container,
|
|
lng_settings_show_sessions,
|
|
st::settingsButton
|
|
)->addClickHandler([] {
|
|
Ui::show(Box<SessionsBox>());
|
|
});
|
|
AddSkip(container, st::settingsPrivacySecurityPadding);
|
|
AddDividerText(
|
|
container,
|
|
Lang::Viewer(lng_settings_sessions_about));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
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);
|
|
}
|
|
|
|
void AddPrivacyButton(
|
|
not_null<Ui::VerticalLayout*> container,
|
|
LangKey label,
|
|
Privacy::Key key,
|
|
Fn<std::unique_ptr<EditPrivacyController>()> controller) {
|
|
const auto shower = Ui::CreateChild<rpl::lifetime>(container.get());
|
|
AddButtonWithLabel(
|
|
container,
|
|
label,
|
|
PrivacyString(key),
|
|
st::settingsButton
|
|
)->addClickHandler([=] {
|
|
*shower = Auth().api().privacyValue(
|
|
key
|
|
) | rpl::take(
|
|
1
|
|
) | rpl::start_with_next([=](const Privacy &value) {
|
|
Ui::show(
|
|
Box<EditPrivacyBox>(controller(), value),
|
|
LayerOption::KeepOther);
|
|
});
|
|
});
|
|
}
|
|
|
|
PrivacySecurity::PrivacySecurity(QWidget *parent, not_null<UserData*> self)
|
|
: Section(parent)
|
|
, _self(self) {
|
|
setupContent();
|
|
}
|
|
|
|
void PrivacySecurity::setupContent() {
|
|
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
|
|
|
|
SetupPrivacy(content);
|
|
SetupSessionsList(content);
|
|
SetupLocalPasscode(content);
|
|
SetupCloudPassword(content);
|
|
SetupSelfDestruction(content);
|
|
|
|
Ui::ResizeFitChild(this, content);
|
|
}
|
|
|
|
} // namespace Settings
|