/* This file is part of jellything (https://codeberg.org/metamuffin/jellything) which is licensed under the GNU Affero General Public License (version 3); see /COPYING. Copyright (C) 2025 metamuffin */ use super::{format_form_error, hash_password}; use crate::{database::Database, locale::AcceptLanguage, ui::error::MyResult}; use jellybase::permission::PermissionSetExt; use jellycommon::user::{PlayerKind, Theme, UserPermission}; use jellylogic::session::Session; use jellyui::locale::Language; use rocket::{ form::{self, validate::len, Contextual, Form}, get, http::uri::fmt::{Query, UriDisplay}, post, response::content::RawHtml, FromForm, State, }; use std::ops::Range; #[derive(FromForm)] pub struct SettingsForm { #[field(validate = option_len(4..64))] password: Option, #[field(validate = option_len(4..32))] display_name: Option, theme: Option, player_preference: Option, native_secret: Option, } fn option_len<'v>(value: &Option, range: Range) -> form::Result<'v, ()> { value.as_ref().map(|v| len(v, range)).unwrap_or(Ok(())) } fn settings_page( session: Session, flash: Option>, lang: Language, ) -> RawHtml { } #[get("/account/settings")] pub fn r_account_settings(session: Session, lang: AcceptLanguage) -> RawHtml { let AcceptLanguage(lang) = lang; settings_page(session, None, lang) } #[post("/account/settings", data = "
")] pub fn r_account_settings_post( session: Session, database: &State, form: Form>, lang: AcceptLanguage, ) -> MyResult> { let AcceptLanguage(lang) = lang; session .user .permissions .assert(&UserPermission::ManageSelf)?; let form = match &form.value { Some(v) => v, None => { return Ok(settings_page( session, Some(Err(format_form_error(form))), lang, )) } }; let mut out = String::new(); database.update_user(&session.user.name, |user| { if let Some(password) = &form.password { user.password = hash_password(&session.user.name, password); out += &*tr(lang, "settings.account.password.changed"); out += "\n"; } if let Some(display_name) = &form.display_name { user.display_name = display_name.clone(); out += &*tr(lang, "settings.account.display_name.changed"); out += "\n"; } if let Some(theme) = form.theme { user.theme = theme; out += &*tr(lang, "settings.account.theme.changed"); out += "\n"; } if let Some(player_preference) = form.player_preference { user.player_preference = player_preference; out += &*tr(lang, "settings.player_preference.changed"); out += "\n"; } if let Some(native_secret) = &form.native_secret { user.native_secret = native_secret.to_owned(); out += "Native secret updated.\n"; } Ok(()) })?; Ok(settings_page( session, // using the old session here, results in outdated theme being displayed Some(Ok(if out.is_empty() { tr(lang, "settings.no_change").to_string() } else { out })), lang, )) }