aboutsummaryrefslogtreecommitdiff
path: root/server/src/ui/account/settings.rs
blob: e6d096f7e139c2e83ae7bf6cc0488e5eb36dc757 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
/*
    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 <metamuffin.org>
*/
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<String>,
    #[field(validate = option_len(4..32))]
    display_name: Option<String>,
    theme: Option<Theme>,
    player_preference: Option<PlayerKind>,
    native_secret: Option<String>,
}

fn option_len<'v>(value: &Option<String>, range: Range<usize>) -> form::Result<'v, ()> {
    value.as_ref().map(|v| len(v, range)).unwrap_or(Ok(()))
}

fn settings_page(
    session: Session,
    flash: Option<MyResult<String>>,
    lang: Language,
) -> RawHtml<String> {
}

#[get("/account/settings")]
pub fn r_account_settings(session: Session, lang: AcceptLanguage) -> RawHtml<String> {
    let AcceptLanguage(lang) = lang;
    settings_page(session, None, lang)
}

#[post("/account/settings", data = "<form>")]
pub fn r_account_settings_post(
    session: Session,
    database: &State<Database>,
    form: Form<Contextual<SettingsForm>>,
    lang: AcceptLanguage,
) -> MyResult<RawHtml<String>> {
    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,
    ))
}