init
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Close stale issues and PRs / stale (push) Has been cancelled
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Close stale issues and PRs / stale (push) Has been cancelled
This commit is contained in:
415
Telegram/SourceFiles/settings/settings_passkeys.cpp
Normal file
415
Telegram/SourceFiles/settings/settings_passkeys.cpp
Normal file
@@ -0,0 +1,415 @@
|
||||
/*
|
||||
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_passkeys.h"
|
||||
|
||||
#include "settings/cloud_password/settings_cloud_password_common.h"
|
||||
#include "settings/settings_common_session.h"
|
||||
#include "data/components/passkeys.h"
|
||||
#include "data/data_session.h"
|
||||
#include "window/window_session_controller.h"
|
||||
#include "main/main_session.h"
|
||||
#include "platform/platform_webauthn.h"
|
||||
#include "ui/layers/generic_box.h"
|
||||
#include "ui/boxes/confirm_box.h"
|
||||
#include "ui/painter.h"
|
||||
#include "ui/rect.h"
|
||||
#include "ui/text/text_utilities.h"
|
||||
#include "ui/vertical_list.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/widgets/popup_menu.h"
|
||||
#include "ui/widgets/menu/menu_add_action_callback.h"
|
||||
#include "ui/widgets/menu/menu_add_action_callback_factory.h"
|
||||
#include "ui/wrap/vertical_layout.h"
|
||||
#include "ui/wrap/slide_wrap.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "lottie/lottie_icon.h"
|
||||
#include "base/unixtime.h"
|
||||
#include "data/stickers/data_custom_emoji.h"
|
||||
#include "ui/text/custom_emoji_instance.h"
|
||||
#include "styles/style_boxes.h"
|
||||
#include "styles/style_channel_earn.h"
|
||||
#include "styles/style_chat.h"
|
||||
#include "styles/style_layers.h"
|
||||
#include "styles/style_premium.h"
|
||||
#include "styles/style_settings.h"
|
||||
#include "styles/style_menu_icons.h"
|
||||
|
||||
namespace Settings {
|
||||
|
||||
class Passkeys : public Section<Passkeys> {
|
||||
public:
|
||||
Passkeys(
|
||||
QWidget *parent,
|
||||
not_null<Window::SessionController*> controller);
|
||||
|
||||
[[nodiscard]] rpl::producer<QString> title() override;
|
||||
|
||||
[[nodiscard]] rpl::producer<> showFinishes() const {
|
||||
return _showFinished.events();
|
||||
}
|
||||
|
||||
const Ui::RoundRect *bottomSkipRounding() const override {
|
||||
return &_bottomSkipRounding;
|
||||
}
|
||||
|
||||
private:
|
||||
void showFinished() override {
|
||||
_showFinished.fire({});
|
||||
}
|
||||
|
||||
void setupContent(not_null<Window::SessionController*> controller);
|
||||
|
||||
Ui::RoundRect _bottomSkipRounding;
|
||||
|
||||
rpl::event_stream<> _showFinished;
|
||||
|
||||
};
|
||||
|
||||
void PasskeysNoneBox(
|
||||
not_null<Ui::GenericBox*> box,
|
||||
not_null<Main::Session*> session) {
|
||||
box->setWidth(st::boxWideWidth);
|
||||
box->setNoContentMargin(true);
|
||||
box->setCloseByEscape(true);
|
||||
box->addTopButton(st::boxTitleClose, [=] { box->closeBox(); });
|
||||
|
||||
const auto content = box->verticalLayout().get();
|
||||
|
||||
Ui::AddSkip(content);
|
||||
{
|
||||
const auto &size = st::settingsCloudPasswordIconSize;
|
||||
auto icon = CreateLottieIcon(
|
||||
content,
|
||||
{ .name = u"passkeys"_q, .sizeOverride = { size, size } },
|
||||
st::settingLocalPasscodeIconPadding);
|
||||
const auto animate = std::move(icon.animate);
|
||||
box->addRow(std::move(icon.widget), style::al_top);
|
||||
box->showFinishes() | rpl::take(1) | rpl::on_next([=] {
|
||||
animate(anim::repeat::once);
|
||||
}, content->lifetime());
|
||||
}
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
box->addRow(
|
||||
object_ptr<Ui::FlatLabel>(
|
||||
content,
|
||||
tr::lng_settings_passkeys_none_title(),
|
||||
st::boxTitle),
|
||||
style::al_top);
|
||||
Ui::AddSkip(content);
|
||||
box->addRow(
|
||||
object_ptr<Ui::FlatLabel>(
|
||||
content,
|
||||
tr::lng_settings_passkeys_none_about(),
|
||||
st::channelEarnLearnDescription),
|
||||
style::al_top);
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
{
|
||||
const auto padding = QMargins(
|
||||
st::settingsButton.padding.left(),
|
||||
st::boxRowPadding.top(),
|
||||
st::boxRowPadding.right(),
|
||||
st::boxRowPadding.bottom());
|
||||
const auto iconLeft = st::settingsButton.iconLeft;
|
||||
const auto addEntry = [&](
|
||||
rpl::producer<QString> title,
|
||||
rpl::producer<QString> about,
|
||||
const style::icon &icon) {
|
||||
const auto top = content->add(
|
||||
object_ptr<Ui::FlatLabel>(
|
||||
content,
|
||||
std::move(title),
|
||||
st::channelEarnSemiboldLabel),
|
||||
padding);
|
||||
Ui::AddSkip(content, st::channelEarnHistoryThreeSkip);
|
||||
content->add(
|
||||
object_ptr<Ui::FlatLabel>(
|
||||
content,
|
||||
std::move(about),
|
||||
st::channelEarnHistoryRecipientLabel),
|
||||
padding);
|
||||
const auto left = Ui::CreateChild<Ui::RpWidget>(
|
||||
box->verticalLayout().get());
|
||||
left->paintRequest(
|
||||
) | rpl::on_next([=] {
|
||||
auto p = Painter(left);
|
||||
icon.paint(p, 0, 0, left->width());
|
||||
}, left->lifetime());
|
||||
left->resize(icon.size());
|
||||
top->geometryValue(
|
||||
) | rpl::on_next([=](const QRect &g) {
|
||||
left->moveToLeft(
|
||||
iconLeft,
|
||||
g.top() + st::channelEarnHistoryThreeSkip);
|
||||
}, left->lifetime());
|
||||
};
|
||||
addEntry(
|
||||
tr::lng_settings_passkeys_none_info1_title(),
|
||||
tr::lng_settings_passkeys_none_info1_about(),
|
||||
st::settingsIconPasskeysAboutIcon1);
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
addEntry(
|
||||
tr::lng_settings_passkeys_none_info2_title(),
|
||||
tr::lng_settings_passkeys_none_info2_about(),
|
||||
st::settingsIconPasskeysAboutIcon2);
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
addEntry(
|
||||
tr::lng_settings_passkeys_none_info3_title(),
|
||||
tr::lng_settings_passkeys_none_info3_about(),
|
||||
st::settingsIconPasskeysAboutIcon3);
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
}
|
||||
Ui::AddSkip(content);
|
||||
Ui::AddSkip(content);
|
||||
{
|
||||
const auto &st = st::premiumPreviewDoubledLimitsBox;
|
||||
const auto canRegister = session->passkeys().canRegister();
|
||||
box->setStyle(st);
|
||||
auto button = object_ptr<Ui::RoundButton>(
|
||||
box,
|
||||
canRegister
|
||||
? tr::lng_settings_passkeys_none_button()
|
||||
: tr::lng_settings_passkeys_none_button_unsupported(),
|
||||
st::defaultActiveButton);
|
||||
button->setTextTransform(Ui::RoundButton::TextTransform::NoTransform);
|
||||
button->resizeToWidth(box->width()
|
||||
- st.buttonPadding.left()
|
||||
- st.buttonPadding.left());
|
||||
button->setClickedCallback([=] {
|
||||
session->passkeys().initRegistration([=](
|
||||
const Data::Passkey::RegisterData &data) {
|
||||
Platform::WebAuthn::RegisterKey(data, [=](
|
||||
Platform::WebAuthn::RegisterResult result) {
|
||||
if (!result.success) {
|
||||
using Error = Platform::WebAuthn::Error;
|
||||
if (result.error == Error::UnsignedBuild) {
|
||||
box->uiShow()->showToast(
|
||||
tr::lng_settings_passkeys_unsigned_error(
|
||||
tr::now));
|
||||
}
|
||||
return;
|
||||
}
|
||||
session->passkeys().registerPasskey(result, [=] {
|
||||
box->closeBox();
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
if (!canRegister) {
|
||||
button->setAttribute(Qt::WA_TransparentForMouseEvents);
|
||||
button->setTextFgOverride(
|
||||
anim::with_alpha(button->st().textFg->c, 0.5));
|
||||
}
|
||||
box->addButton(std::move(button));
|
||||
}
|
||||
}
|
||||
|
||||
Passkeys::Passkeys(
|
||||
QWidget *parent,
|
||||
not_null<Window::SessionController*> controller)
|
||||
: Section(parent)
|
||||
, _bottomSkipRounding(st::boxRadius, st::boxDividerBg) {
|
||||
setupContent(controller);
|
||||
}
|
||||
|
||||
rpl::producer<QString> Passkeys::title() {
|
||||
return tr::lng_settings_passkeys_title();
|
||||
}
|
||||
|
||||
void Passkeys::setupContent(
|
||||
not_null<Window::SessionController*> controller) {
|
||||
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
|
||||
const auto session = &controller->session();
|
||||
|
||||
CloudPassword::SetupHeader(
|
||||
content,
|
||||
u"passkeys"_q,
|
||||
showFinishes(),
|
||||
rpl::single(QString()),
|
||||
tr::lng_settings_passkeys_about());
|
||||
|
||||
Ui::AddSkip(content);
|
||||
|
||||
const auto container = content->add(
|
||||
object_ptr<Ui::VerticalLayout>(content));
|
||||
|
||||
const auto &st = st::peerListBoxItem;
|
||||
const auto nameStyle = &st.nameStyle;
|
||||
const auto rebuild = [=] {
|
||||
while (container->count()) {
|
||||
delete container->widgetAt(0);
|
||||
}
|
||||
for (const auto &passkey : session->passkeys().list()) {
|
||||
const auto button = container->add(
|
||||
object_ptr<Ui::AbstractButton>(container));
|
||||
button->resize(button->width(), st.height);
|
||||
const auto menu = Ui::CreateChild<Ui::IconButton>(
|
||||
button,
|
||||
st::themesMenuToggle);
|
||||
menu->setClickedCallback([=] {
|
||||
const auto popup = Ui::CreateChild<Ui::PopupMenu>(
|
||||
menu,
|
||||
st::popupMenuWithIcons);
|
||||
const auto handler = [=, id = passkey.id] {
|
||||
controller->show(Ui::MakeConfirmBox({
|
||||
.text = rpl::combine(
|
||||
tr::lng_settings_passkeys_delete_sure_about(),
|
||||
tr::lng_settings_passkeys_delete_sure_about2()
|
||||
) | rpl::map([](QString a, QString b) {
|
||||
return a + "\n\n" + b;
|
||||
}),
|
||||
.confirmed = [=](Fn<void()> close) {
|
||||
session->passkeys().deletePasskey(
|
||||
id,
|
||||
close,
|
||||
[](QString) {});
|
||||
},
|
||||
.confirmText = tr::lng_box_delete(),
|
||||
.confirmStyle = &st::attentionBoxButton,
|
||||
.title
|
||||
= tr::lng_settings_passkeys_delete_sure_title(),
|
||||
}));
|
||||
};
|
||||
Ui::Menu::CreateAddActionCallback(popup)({
|
||||
.text = tr::lng_proxy_menu_delete(tr::now),
|
||||
.handler = handler,
|
||||
.icon = &st::menuIconDeleteAttention,
|
||||
.isAttention = true,
|
||||
});
|
||||
popup->setForcedOrigin(Ui::PanelAnimation::Origin::TopRight);
|
||||
const auto menuGlobal = menu->mapToGlobal(
|
||||
QPoint(menu->width(), menu->height()));
|
||||
popup->popup(menuGlobal);
|
||||
});
|
||||
button->widthValue() | rpl::on_next([=](int width) {
|
||||
menu->moveToRight(0, (st.height - menu->height()) / 2, width);
|
||||
}, button->lifetime());
|
||||
const auto iconSize = st::settingsIconPasskeys.width();
|
||||
const auto emoji = iconSize;
|
||||
const auto iconLeft = st::settingsButton.iconLeft;
|
||||
auto emojiInstance = passkey.softwareEmojiId
|
||||
? session->data().customEmojiManager().create(
|
||||
passkey.softwareEmojiId,
|
||||
[=] { button->update(); },
|
||||
Data::CustomEmojiSizeTag::Large,
|
||||
emoji)
|
||||
: nullptr;
|
||||
const auto emojiPtr = emojiInstance.get();
|
||||
button->lifetime().add([emoji = std::move(emojiInstance)] {});
|
||||
const auto formatDateTime = [](TimeId timestamp) {
|
||||
const auto dt = base::unixtime::parse(timestamp);
|
||||
return tr::lng_mediaview_date_time(
|
||||
tr::now,
|
||||
lt_date,
|
||||
langDayOfMonthFull(dt.date()),
|
||||
lt_time,
|
||||
QLocale().toString(dt.time(), QLocale::ShortFormat));
|
||||
};
|
||||
const auto date = (passkey.lastUsageDate > 0)
|
||||
? tr::lng_settings_passkeys_last_used(
|
||||
tr::now,
|
||||
lt_date,
|
||||
formatDateTime(passkey.lastUsageDate))
|
||||
: tr::lng_settings_passkeys_created(
|
||||
tr::now,
|
||||
lt_date,
|
||||
formatDateTime(passkey.date));
|
||||
const auto nameText = button->lifetime().make_state<
|
||||
Ui::Text::String>(
|
||||
*nameStyle,
|
||||
passkey.name.isEmpty()
|
||||
? tr::lng_settings_passkey_unknown(tr::now)
|
||||
: passkey.name);
|
||||
const auto dateText = button->lifetime().make_state<
|
||||
Ui::Text::String>(st::defaultTextStyle, date);
|
||||
button->paintOn([=](QPainter &p) {
|
||||
const auto iconTop = (st.height - iconSize) / 2;
|
||||
if (emojiPtr) {
|
||||
emojiPtr->paint(p, {
|
||||
.textColor = st.nameFg->c,
|
||||
.now = crl::now(),
|
||||
.position = QPoint(iconLeft, iconTop),
|
||||
});
|
||||
} else {
|
||||
const auto w = button->width();
|
||||
st::settingsIconPasskeys.paint(p, iconLeft, iconTop, w);
|
||||
}
|
||||
const auto textLeft = st::settingsButton.padding.left();
|
||||
const auto textWidth = button->width() - textLeft
|
||||
- st::settingsButton.padding.right();
|
||||
p.setPen(st.nameFg);
|
||||
nameText->draw(p, {
|
||||
.position = { textLeft, st.namePosition.y() },
|
||||
.outerWidth = button->width(),
|
||||
.availableWidth = textWidth,
|
||||
.elisionLines = 1,
|
||||
});
|
||||
p.setPen(st.statusFg);
|
||||
dateText->draw(p, {
|
||||
.position = { textLeft, st.statusPosition.y() },
|
||||
.outerWidth = button->width(),
|
||||
.availableWidth = textWidth,
|
||||
.elisionLines = 1,
|
||||
});
|
||||
});
|
||||
button->showChildren();
|
||||
}
|
||||
container->showChildren();
|
||||
container->resizeToWidth(content->width());
|
||||
};
|
||||
|
||||
const auto buttonWrap = content->add(
|
||||
object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
|
||||
content,
|
||||
CreateButtonWithIcon(
|
||||
content,
|
||||
tr::lng_settings_passkeys_button(),
|
||||
st::settingsButtonActive,
|
||||
{ &st::settingsIconPasskeys })));
|
||||
buttonWrap->entity()->setClickedCallback([=] {
|
||||
controller->show(Box(PasskeysNoneBox, session));
|
||||
});
|
||||
buttonWrap->toggleOn(session->passkeys().requestList(
|
||||
) | rpl::map([=] { return session->passkeys().canRegister(); }));
|
||||
buttonWrap->finishAnimating();
|
||||
|
||||
session->passkeys().requestList(
|
||||
) | rpl::on_next(rebuild, content->lifetime());
|
||||
rebuild();
|
||||
|
||||
Ui::AddSkip(content);
|
||||
const auto label = Ui::AddDividerText(
|
||||
content,
|
||||
tr::lng_settings_passkeys_button_about(
|
||||
lt_link,
|
||||
tr::lng_channel_earn_about_link(
|
||||
lt_emoji,
|
||||
rpl::single(Ui::Text::IconEmoji(&st::textMoreIconEmoji)),
|
||||
tr::rich
|
||||
) | rpl::map([](TextWithEntities text) {
|
||||
return tr::link(std::move(text), u"internal"_q);
|
||||
}),
|
||||
tr::rich
|
||||
));
|
||||
label->setClickHandlerFilter([=](const auto &...) {
|
||||
controller->show(Box(PasskeysNoneBox, session));
|
||||
return false;
|
||||
});
|
||||
Ui::ResizeFitChild(this, content);
|
||||
}
|
||||
|
||||
Type PasskeysId() {
|
||||
return Passkeys::Id();
|
||||
}
|
||||
|
||||
} // namespace Settings
|
||||
Reference in New Issue
Block a user