fractal/session/view/account_settings/general_page/
change_password_subpage.rsuse adw::{prelude::*, subclass::prelude::*};
use gettextrs::gettext;
use gtk::{
glib::{self, clone},
CompositeTemplate,
};
use matrix_sdk::ruma::{
api::client::{account::change_password, error::ErrorKind},
assign,
};
use tracing::error;
use crate::{
components::{AuthDialog, AuthError, LoadingButtonRow},
session::model::Session,
toast,
utils::matrix::validate_password,
};
mod imp {
use glib::subclass::InitializingObject;
use super::*;
#[derive(Debug, Default, CompositeTemplate, glib::Properties)]
#[template(
resource = "/org/gnome/Fractal/ui/session/view/account_settings/general_page/change_password_subpage.ui"
)]
#[properties(wrapper_type = super::ChangePasswordSubpage)]
pub struct ChangePasswordSubpage {
#[property(get, set, nullable)]
pub session: glib::WeakRef<Session>,
#[template_child]
pub password: TemplateChild<adw::PasswordEntryRow>,
#[template_child]
pub password_progress: TemplateChild<gtk::LevelBar>,
#[template_child]
pub password_error_revealer: TemplateChild<gtk::Revealer>,
#[template_child]
pub password_error: TemplateChild<gtk::Label>,
#[template_child]
pub confirm_password: TemplateChild<adw::PasswordEntryRow>,
#[template_child]
pub confirm_password_error_revealer: TemplateChild<gtk::Revealer>,
#[template_child]
pub confirm_password_error: TemplateChild<gtk::Label>,
#[template_child]
pub button: TemplateChild<LoadingButtonRow>,
}
#[glib::object_subclass]
impl ObjectSubclass for ChangePasswordSubpage {
const NAME: &'static str = "ChangePasswordSubpage";
type Type = super::ChangePasswordSubpage;
type ParentType = adw::NavigationPage;
fn class_init(klass: &mut Self::Class) {
Self::bind_template(klass);
Self::Type::bind_template_callbacks(klass);
}
fn instance_init(obj: &InitializingObject<Self>) {
obj.init_template();
}
}
#[glib::derived_properties]
impl ObjectImpl for ChangePasswordSubpage {
fn constructed(&self) {
self.parent_constructed();
let obj = self.obj();
self.password_progress.set_min_value(0.0);
self.password_progress.set_max_value(5.0);
self.password_progress
.add_offset_value(gtk::LEVEL_BAR_OFFSET_LOW, 1.0);
self.password_progress.add_offset_value("step2", 2.0);
self.password_progress.add_offset_value("step3", 3.0);
self.password_progress
.add_offset_value(gtk::LEVEL_BAR_OFFSET_HIGH, 4.0);
self.password_progress
.add_offset_value(gtk::LEVEL_BAR_OFFSET_FULL, 5.0);
self.password.connect_changed(clone!(
#[weak]
obj,
move |_| {
obj.validate_password();
obj.validate_password_confirmation();
}
));
self.confirm_password.connect_changed(clone!(
#[weak]
obj,
move |_| {
obj.validate_password_confirmation();
}
));
}
}
impl WidgetImpl for ChangePasswordSubpage {}
impl NavigationPageImpl for ChangePasswordSubpage {}
}
glib::wrapper! {
pub struct ChangePasswordSubpage(ObjectSubclass<imp::ChangePasswordSubpage>)
@extends gtk::Widget, adw::NavigationPage, @implements gtk::Accessible;
}
#[gtk::template_callbacks]
impl ChangePasswordSubpage {
pub fn new(session: &Session) -> Self {
glib::Object::builder().property("session", session).build()
}
fn validate_password(&self) {
let imp = self.imp();
let entry = &imp.password;
let progress = &imp.password_progress;
let revealer = &imp.password_error_revealer;
let label = &imp.password_error;
let password = entry.text();
if password.is_empty() {
revealer.set_reveal_child(false);
entry.remove_css_class("success");
entry.remove_css_class("warning");
progress.set_value(0.0);
progress.remove_css_class("success");
progress.remove_css_class("warning");
self.update_button();
return;
}
let validity = validate_password(&password);
progress.set_value(f64::from(validity.progress) / 20.0);
if validity.progress == 100 {
revealer.set_reveal_child(false);
entry.add_css_class("success");
entry.remove_css_class("warning");
progress.add_css_class("success");
progress.remove_css_class("warning");
} else {
entry.remove_css_class("success");
entry.add_css_class("warning");
progress.remove_css_class("success");
progress.add_css_class("warning");
if !validity.has_length {
label.set_label(&gettext("Password must be at least 8 characters long"));
} else if !validity.has_lowercase {
label.set_label(&gettext(
"Password must have at least one lower-case letter",
));
} else if !validity.has_uppercase {
label.set_label(&gettext(
"Password must have at least one upper-case letter",
));
} else if !validity.has_number {
label.set_label(&gettext("Password must have at least one digit"));
} else if !validity.has_symbol {
label.set_label(&gettext("Password must have at least one symbol"));
}
revealer.set_reveal_child(true);
}
self.update_button();
}
fn validate_password_confirmation(&self) {
let imp = self.imp();
let entry = &imp.confirm_password;
let revealer = &imp.confirm_password_error_revealer;
let label = &imp.confirm_password_error;
let password = imp.password.text();
let confirmation = entry.text();
if confirmation.is_empty() {
revealer.set_reveal_child(false);
entry.remove_css_class("success");
entry.remove_css_class("warning");
return;
}
if password == confirmation {
revealer.set_reveal_child(false);
entry.add_css_class("success");
entry.remove_css_class("warning");
} else {
entry.remove_css_class("success");
entry.add_css_class("warning");
label.set_label(&gettext("Passwords do not match"));
revealer.set_reveal_child(true);
}
self.update_button();
}
fn update_button(&self) {
self.imp().button.set_sensitive(self.can_change_password());
}
fn can_change_password(&self) -> bool {
let imp = self.imp();
let password = imp.password.text();
let confirmation = imp.confirm_password.text();
validate_password(&password).progress == 100 && password == confirmation
}
#[template_callback]
async fn change_password(&self) {
let Some(session) = self.session() else {
return;
};
if !self.can_change_password() {
return;
}
let imp = self.imp();
let password = imp.password.text();
imp.button.set_is_loading(true);
imp.password.set_sensitive(false);
imp.confirm_password.set_sensitive(false);
let dialog = AuthDialog::new(&session);
let result = dialog
.authenticate(self, move |client, auth| {
let password = password.clone();
async move {
let request =
assign!(change_password::v3::Request::new(password.into()), { auth });
client.send(request, None).await.map_err(Into::into)
}
})
.await;
match result {
Ok(_) => {
toast!(self, gettext("Password changed successfully"));
imp.password.set_text("");
imp.confirm_password.set_text("");
self.activate_action("account-settings.close-subpage", None)
.unwrap();
}
Err(error) => match error {
AuthError::UserCancelled => {}
AuthError::ServerResponse(error)
if matches!(error.client_api_error_kind(), Some(ErrorKind::WeakPassword)) =>
{
error!("Weak password: {error}");
toast!(self, gettext("Password rejected for being too weak"));
}
_ => {
error!("Could not change the password: {error:?}");
toast!(self, gettext("Could not change password"));
}
},
}
imp.button.set_is_loading(false);
imp.password.set_sensitive(true);
imp.confirm_password.set_sensitive(true);
}
}