single-user-edit-page #970
@@ -50,11 +50,16 @@ function editReadOnlyField() {
 | 
			
		||||
      Array.prototype.forEach.call(editBtns, (btn: HTMLButtonElement) => {
 | 
			
		||||
        btn.addEventListener("click", function () {
 | 
			
		||||
          let wrapper = btn.parentElement;
 | 
			
		||||
          let input = wrapper?.querySelector('input');
 | 
			
		||||
          let input = <HTMLInputElement> wrapper?.querySelector('input.input'),
 | 
			
		||||
              select = <HTMLSelectElement> wrapper?.querySelector('select.input'),
 | 
			
		||||
              attribute = 'readonly';
 | 
			
		||||
          
 | 
			
		||||
          if(select) attribute = 'disabled';
 | 
			
		||||
          let element = input ? input : select;
 | 
			
		||||
 | 
			
		||||
          wrapper?.classList.toggle('editable')
 | 
			
		||||
          input?.toggleAttribute('readonly');
 | 
			
		||||
          if(!input?.hasAttribute('readonly')) input?.focus();
 | 
			
		||||
          element?.toggleAttribute(attribute);
 | 
			
		||||
          if(!element?.hasAttribute(attribute)) element?.focus();
 | 
			
		||||
          wrapper?.classList.toggle('editable');
 | 
			
		||||
        });
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										70
									
								
								src/lib.rs
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								src/lib.rs
									
									
									
									
									
								
							@@ -1,5 +1,7 @@
 | 
			
		||||
#![allow(clippy::blocks_in_conditions)]
 | 
			
		||||
 | 
			
		||||
use std::ops::Deref;
 | 
			
		||||
 | 
			
		||||
pub mod model;
 | 
			
		||||
 | 
			
		||||
#[cfg(feature = "rowing-tera")]
 | 
			
		||||
@@ -22,6 +24,74 @@ pub(crate) const FOERDERND: i64 = 8500;
 | 
			
		||||
pub(crate) const SCHECKBUCH: i64 = 3000;
 | 
			
		||||
pub(crate) const EINSCHREIBGEBUEHR: i64 = 3000;
 | 
			
		||||
 | 
			
		||||
#[derive(Debug, Clone, PartialEq, Eq)]
 | 
			
		||||
pub struct NonEmptyString(String);
 | 
			
		||||
 | 
			
		||||
impl NonEmptyString {
 | 
			
		||||
    pub fn new(s: String) -> Option<Self> {
 | 
			
		||||
        if s.is_empty() {
 | 
			
		||||
            None
 | 
			
		||||
        } else {
 | 
			
		||||
            Some(NonEmptyString(s))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn as_str(&self) -> &str {
 | 
			
		||||
        &self.0
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn into_string(self) -> String {
 | 
			
		||||
        self.0
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Implement Deref to allow automatic dereferencing to &str
 | 
			
		||||
impl Deref for NonEmptyString {
 | 
			
		||||
    type Target = str;
 | 
			
		||||
 | 
			
		||||
    fn deref(&self) -> &Self::Target {
 | 
			
		||||
        &self.0
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// This allows &NonEmptyString to be converted to &str
 | 
			
		||||
impl AsRef<str> for NonEmptyString {
 | 
			
		||||
    fn as_ref(&self) -> &str {
 | 
			
		||||
        &self.0
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// This allows NonEmptyString to be converted to String with .into()
 | 
			
		||||
impl From<NonEmptyString> for String {
 | 
			
		||||
    fn from(s: NonEmptyString) -> Self {
 | 
			
		||||
        s.0
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl TryFrom<&str> for NonEmptyString {
 | 
			
		||||
    type Error = &'static str;
 | 
			
		||||
 | 
			
		||||
    fn try_from(s: &str) -> Result<Self, Self::Error> {
 | 
			
		||||
        if s.is_empty() {
 | 
			
		||||
            Err("String cannot be empty")
 | 
			
		||||
        } else {
 | 
			
		||||
            Ok(NonEmptyString(s.to_string()))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl TryFrom<String> for NonEmptyString {
 | 
			
		||||
    type Error = &'static str;
 | 
			
		||||
 | 
			
		||||
    fn try_from(s: String) -> Result<Self, Self::Error> {
 | 
			
		||||
        if s.is_empty() {
 | 
			
		||||
            Err("String cannot be empty")
 | 
			
		||||
        } else {
 | 
			
		||||
            Ok(NonEmptyString(s))
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[cfg(test)]
 | 
			
		||||
#[macro_export]
 | 
			
		||||
macro_rules! testdb {
 | 
			
		||||
 
 | 
			
		||||
@@ -42,12 +42,20 @@ impl User {
 | 
			
		||||
        db: &SqlitePool,
 | 
			
		||||
        updated_by: &ManageUserUser,
 | 
			
		||||
        new_phone: &str,
 | 
			
		||||
    ) -> Result<(), String> {
 | 
			
		||||
    ) {
 | 
			
		||||
        let new_phone = new_phone.trim();
 | 
			
		||||
 | 
			
		||||
        let query = if new_phone.is_empty() {
 | 
			
		||||
            if self.phone.is_none() {
 | 
			
		||||
                return; // nothing to do
 | 
			
		||||
            }
 | 
			
		||||
            sqlx::query!("UPDATE user SET phone = NULL where id = ?", self.id)
 | 
			
		||||
        } else {
 | 
			
		||||
            if let Some(old_phone) = &self.phone {
 | 
			
		||||
                if old_phone == new_phone {
 | 
			
		||||
                    return; //nothing to do
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            sqlx::query!("UPDATE user SET phone = ? where id = ?", new_phone, self.id)
 | 
			
		||||
        };
 | 
			
		||||
        query.execute(db).await.unwrap(); //Okay, because we can only create a User of a valid id
 | 
			
		||||
@@ -58,8 +66,6 @@ impl User {
 | 
			
		||||
            None => format!("{updated_by} has added a phone number for {self}: {new_phone}")
 | 
			
		||||
        };
 | 
			
		||||
        Log::create(db, msg).await;
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub(crate) async fn update_address(
 | 
			
		||||
@@ -67,12 +73,20 @@ impl User {
 | 
			
		||||
        db: &SqlitePool,
 | 
			
		||||
        updated_by: &ManageUserUser,
 | 
			
		||||
        new_address: &str,
 | 
			
		||||
    ) -> Result<(), String> {
 | 
			
		||||
    ) {
 | 
			
		||||
        let new_address = new_address.trim();
 | 
			
		||||
 | 
			
		||||
        let query = if new_address.is_empty() {
 | 
			
		||||
            if !self.address.is_none() {
 | 
			
		||||
                return; // nothing to do
 | 
			
		||||
            }
 | 
			
		||||
            sqlx::query!("UPDATE user SET address = NULL where id = ?", self.id)
 | 
			
		||||
        } else {
 | 
			
		||||
            if let Some(old_address) = &self.address {
 | 
			
		||||
                if old_address == new_address {
 | 
			
		||||
                    return; //nothing to do
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            sqlx::query!(
 | 
			
		||||
                "UPDATE user SET address = ? where id = ?",
 | 
			
		||||
                new_address,
 | 
			
		||||
@@ -87,8 +101,6 @@ impl User {
 | 
			
		||||
            None => format!("{updated_by} has added an address for {self}: {new_address}")
 | 
			
		||||
        };
 | 
			
		||||
        Log::create(db, msg).await;
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub(crate) async fn update_nickname(
 | 
			
		||||
@@ -313,6 +325,9 @@ impl User {
 | 
			
		||||
        if self.has_membership_pdf(db).await {
 | 
			
		||||
            return Err(format!("User {self} hat bereits eine Beitrittserklärung."));
 | 
			
		||||
        }
 | 
			
		||||
        if membership_pdf.len() == 0 {
 | 
			
		||||
            return Err(format!("Keine Beitrittserklärung mitgeschickt."));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        let mut stream = membership_pdf.open().await.unwrap();
 | 
			
		||||
        let mut buffer = Vec::new();
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,6 @@
 | 
			
		||||
use super::ScheckbuchUser;
 | 
			
		||||
use crate::model::{
 | 
			
		||||
    logbook::{Logbook, LogbookWithBoatAndRowers},
 | 
			
		||||
    role::Role,
 | 
			
		||||
    user::User,
 | 
			
		||||
};
 | 
			
		||||
use serde::{Deserialize, Serialize};
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,12 @@
 | 
			
		||||
use std::{
 | 
			
		||||
    fmt::Display,
 | 
			
		||||
    ops::{Deref, DerefMut},
 | 
			
		||||
};
 | 
			
		||||
use std::{fmt::Display, ops::DerefMut};
 | 
			
		||||
 | 
			
		||||
use argon2::{password_hash::SaltString, Argon2, PasswordHasher};
 | 
			
		||||
use chrono::{Datelike, Local, NaiveDate};
 | 
			
		||||
use log::info;
 | 
			
		||||
use rocket::async_trait;
 | 
			
		||||
use rocket::{
 | 
			
		||||
    async_trait,
 | 
			
		||||
    http::{Cookie, Status},
 | 
			
		||||
    request::{self, FromRequest, Outcome},
 | 
			
		||||
    request::{FromRequest, Outcome},
 | 
			
		||||
    time::{Duration, OffsetDateTime},
 | 
			
		||||
    tokio::io::AsyncReadExt,
 | 
			
		||||
    Request,
 | 
			
		||||
@@ -35,6 +32,7 @@ use scheckbuch::ScheckbuchUser;
 | 
			
		||||
mod basic;
 | 
			
		||||
mod fee;
 | 
			
		||||
pub(crate) mod member;
 | 
			
		||||
pub(crate) mod regular;
 | 
			
		||||
pub(crate) mod scheckbuch;
 | 
			
		||||
 | 
			
		||||
#[derive(FromRow, Serialize, Deserialize, Clone, Debug, Eq, Hash, PartialEq)]
 | 
			
		||||
@@ -119,10 +117,7 @@ impl User {
 | 
			
		||||
            ));
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        if self.has_role(db, "Donau Linz").await {
 | 
			
		||||
            self.send_welcome_mail_full_member(db, mail, smtp_pw)
 | 
			
		||||
                .await?;
 | 
			
		||||
        } else if self.has_role(db, "schnupperant").await {
 | 
			
		||||
        if self.has_role(db, "schnupperant").await {
 | 
			
		||||
            self.send_welcome_mail_schnupper(db, mail, smtp_pw).await?;
 | 
			
		||||
        } else if let Some(scheckbuch) = ScheckbuchUser::new(db, self).await {
 | 
			
		||||
            scheckbuch.notify(db, mail, smtp_pw).await?;
 | 
			
		||||
@@ -182,57 +177,6 @@ ASKÖ Ruderverein Donau Linz", self.name),
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async fn send_welcome_mail_full_member(
 | 
			
		||||
        &self,
 | 
			
		||||
        db: &SqlitePool,
 | 
			
		||||
        mail: &str,
 | 
			
		||||
        smtp_pw: &str,
 | 
			
		||||
    ) -> Result<(), String> {
 | 
			
		||||
        // 2 things to do:
 | 
			
		||||
        // 1. Send mail to user
 | 
			
		||||
        Mail::send_single(
 | 
			
		||||
            db,
 | 
			
		||||
            mail,
 | 
			
		||||
            "Willkommen im ASKÖ Ruderverein Donau Linz!",
 | 
			
		||||
            format!(
 | 
			
		||||
"Hallo {0},
 | 
			
		||||
 | 
			
		||||
herzlich willkommen im ASKÖ Ruderverein Donau Linz! Wir freuen uns sehr, dich als neues Mitglied in unserem Verein begrüßen zu dürfen. 
 | 
			
		||||
 | 
			
		||||
Um dir den Einstieg zu erleichtern, findest du in unserem Handbuch alle wichtigen Informationen über unseren Verein: https://rudernlinz.at/book. Bei weiteren Fragen stehen dir die Adressen info@rudernlinz.at (für allgemeine Fragen) und it@rudernlinz.at (bei technischen Fragen) jederzeit zur Verfügung.
 | 
			
		||||
 | 
			
		||||
Du kannst auch gerne unserer Signal-Gruppe beitreten, um auf dem Laufenden zu bleiben und dich mit anderen Mitgliedern auszutauschen: https://signal.group/#CjQKICFrq6zSsRHxrucS3jEcQn6lknEXacAykwwLV3vNLKxPEhA17jxz7cpjfu3JZokLq1TH
 | 
			
		||||
 | 
			
		||||
Für die Organisation unserer Ausfahrten nutzen wir app.rudernlinz.at. Logge dich einfach mit deinem Namen ('{0}' ohne Anführungszeichen) ein, beim ersten Mal kannst du das Passwortfeld leer lassen. Unter 'Geplante Ausfahrten' kannst du dich jederzeit zu den Ausfahrten anmelden.
 | 
			
		||||
 | 
			
		||||
Beim nächsten Treffen im Verein, erinnere jemand vom Vorstand (https://rudernlinz.at/unser-verein/vorstand/) bitte daran, deinen Fingerabdruck zu registrieren, damit du Zugang zum Bootshaus erhältst.
 | 
			
		||||
 | 
			
		||||
Damit du dich noch mehr verbunden fühlst (:-)), haben wir im Bootshaus ein WLAN für Vereinsmitglieder 'ASKÖ Ruderverein Donau Linz' eingerichtet. Das Passwort dafür lautet 'donau1921' (ohne Anführungszeichen). Bitte gib das Passwort an keine vereinsfremden Personen weiter.
 | 
			
		||||
 | 
			
		||||
Wir freuen uns darauf, dich bald am Wasser zu sehen und gemeinsam tolle Erfahrungen zu sammeln!
 | 
			
		||||
 | 
			
		||||
Riemen- & Dollenbruch
 | 
			
		||||
ASKÖ Ruderverein Donau Linz", self.name),
 | 
			
		||||
            smtp_pw,
 | 
			
		||||
        ).await?;
 | 
			
		||||
 | 
			
		||||
        // 2. Notify all coxes
 | 
			
		||||
        Notification::create_for_steering_people(
 | 
			
		||||
            db,
 | 
			
		||||
            &format!(
 | 
			
		||||
                "Liebe Steuerberechtigte, seit {} gibt es ein neues Mitglied: {}",
 | 
			
		||||
                self.member_since_date.clone().unwrap(),
 | 
			
		||||
                self.name
 | 
			
		||||
            ),
 | 
			
		||||
            "Neues Vereinsmitglied",
 | 
			
		||||
            None,
 | 
			
		||||
            None,
 | 
			
		||||
        )
 | 
			
		||||
        .await;
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub async fn amount_boats(&self, db: &SqlitePool) -> i64 {
 | 
			
		||||
        sqlx::query!(
 | 
			
		||||
            "SELECT COUNT(*) as count FROM boat WHERE owner = ?",
 | 
			
		||||
@@ -904,7 +848,7 @@ ASKÖ Ruderverein Donau Linz", self.name),
 | 
			
		||||
impl<'r> FromRequest<'r> for User {
 | 
			
		||||
    type Error = LoginError;
 | 
			
		||||
 | 
			
		||||
    async fn from_request(req: &'r Request<'_>) -> request::Outcome<Self, Self::Error> {
 | 
			
		||||
    async fn from_request(req: &'r Request<'_>) -> rocket::request::Outcome<Self, Self::Error> {
 | 
			
		||||
        match req.cookies().get_private("loggedin_user") {
 | 
			
		||||
            Some(user_id) => match user_id.value().parse::<i32>() {
 | 
			
		||||
                Ok(user_id) => {
 | 
			
		||||
@@ -939,7 +883,7 @@ macro_rules! special_user {
 | 
			
		||||
            pub(crate) user: User,
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        impl Deref for $name {
 | 
			
		||||
        impl std::ops::Deref for $name {
 | 
			
		||||
            type Target = User;
 | 
			
		||||
            fn deref(&self) -> &Self::Target {
 | 
			
		||||
                &self.user
 | 
			
		||||
@@ -953,20 +897,20 @@ macro_rules! special_user {
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #[async_trait]
 | 
			
		||||
        impl<'r> FromRequest<'r> for $name {
 | 
			
		||||
            type Error = LoginError;
 | 
			
		||||
            async fn from_request(req: &'r Request<'_>) -> request::Outcome<Self, Self::Error> {
 | 
			
		||||
        impl<'r> rocket::request::FromRequest<'r> for $name {
 | 
			
		||||
            type Error = crate::model::user::LoginError;
 | 
			
		||||
            async fn from_request(req: &'r rocket::request::Request<'_>) -> rocket::request::Outcome<Self, Self::Error> {
 | 
			
		||||
                let db = req.rocket().state::<SqlitePool>().unwrap();
 | 
			
		||||
                match User::from_request(req).await {
 | 
			
		||||
                    Outcome::Success(user) => {
 | 
			
		||||
                    rocket::request::Outcome::Success(user) => {
 | 
			
		||||
                        if special_user!(@check_roles user, db, $($role)*) {
 | 
			
		||||
                            Outcome::Success($name { user })
 | 
			
		||||
                            rocket::request::Outcome::Success($name { user })
 | 
			
		||||
                        } else {
 | 
			
		||||
                            Outcome::Forward(Status::Forbidden)
 | 
			
		||||
                            rocket::request::Outcome::Forward(rocket::http::Status::Forbidden)
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    Outcome::Error(f) => Outcome::Error(f),
 | 
			
		||||
                    Outcome::Forward(f) => Outcome::Forward(f),
 | 
			
		||||
                    rocket::request::Outcome::Error(f) => rocket::request::Outcome::Error(f),
 | 
			
		||||
                    rocket::request::Outcome::Forward(f) => rocket::request::Outcome::Forward(f),
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -1007,8 +951,10 @@ special_user!(TechUser, +"tech");
 | 
			
		||||
special_user!(ErgoUser, +"ergo");
 | 
			
		||||
special_user!(SteeringUser, +"cox", +"Bootsführer");
 | 
			
		||||
special_user!(AdminUser, +"admin");
 | 
			
		||||
special_user!(AllowedForPlannedTripsUser, +"Donau Linz", +"scheckbuch");
 | 
			
		||||
special_user!(DonauLinzUser, +"Donau Linz", -"Unterstützend", -"Förderndes Mitglied");
 | 
			
		||||
special_user!(AllowedForPlannedTripsUser, +"Donau Linz", +"scheckbuch", +"Förderndes Mitglied");
 | 
			
		||||
special_user!(DonauLinzUser, +"Donau Linz", -"Unterstützend", -"Förderndes Mitglied"); // TODO:
 | 
			
		||||
                                                                                       // remove ->
 | 
			
		||||
                                                                                       // RegularUser
 | 
			
		||||
special_user!(SchnupperBetreuerUser, +"schnupper-betreuer");
 | 
			
		||||
special_user!(VorstandUser, +"admin", +"Vorstand");
 | 
			
		||||
special_user!(EventUser, +"manage_events");
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										73
									
								
								src/model/user/regular.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								src/model/user/regular.rs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,73 @@
 | 
			
		||||
use super::User;
 | 
			
		||||
use crate::{
 | 
			
		||||
    model::{mail::Mail, notification::Notification},
 | 
			
		||||
    special_user,
 | 
			
		||||
};
 | 
			
		||||
use rocket::async_trait;
 | 
			
		||||
use sqlx::SqlitePool;
 | 
			
		||||
 | 
			
		||||
special_user!(RegularUser, +"Donau Linz", -"Unterstützend", -"Förderndes Mitglied");
 | 
			
		||||
 | 
			
		||||
impl RegularUser {
 | 
			
		||||
    pub(crate) async fn notify(&self, db: &SqlitePool, smtp_pw: &str) -> Result<(), String> {
 | 
			
		||||
        self.notify_coxes_about_new_regular(db).await;
 | 
			
		||||
        self.send_welcome_mail_to_user(db, smtp_pw).await?;
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async fn send_welcome_mail_to_user(
 | 
			
		||||
        &self,
 | 
			
		||||
        db: &SqlitePool,
 | 
			
		||||
        smtp_pw: &str,
 | 
			
		||||
    ) -> Result<(), String> {
 | 
			
		||||
        let Some(mail) = &self.mail else {
 | 
			
		||||
            return Err(format!(
 | 
			
		||||
                "Couldn't send welcome mail, as the user {self} has no mail..."
 | 
			
		||||
            ));
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        Mail::send_single(
 | 
			
		||||
            db,
 | 
			
		||||
            mail,
 | 
			
		||||
            "Willkommen im ASKÖ Ruderverein Donau Linz!",
 | 
			
		||||
            format!(
 | 
			
		||||
"Hallo {0},
 | 
			
		||||
 | 
			
		||||
herzlich willkommen im ASKÖ Ruderverein Donau Linz! Wir freuen uns sehr, dich als neues Mitglied in unserem Verein begrüßen zu dürfen. 
 | 
			
		||||
 | 
			
		||||
Um dir den Einstieg zu erleichtern, findest du in unserem Handbuch alle wichtigen Informationen über unseren Verein: https://rudernlinz.at/book. Bei weiteren Fragen stehen dir die Adressen info@rudernlinz.at (für allgemeine Fragen) und it@rudernlinz.at (bei technischen Fragen) jederzeit zur Verfügung.
 | 
			
		||||
 | 
			
		||||
Du kannst auch gerne unserer Signal-Gruppe beitreten, um auf dem Laufenden zu bleiben und dich mit anderen Mitgliedern auszutauschen: https://signal.group/#CjQKICFrq6zSsRHxrucS3jEcQn6lknEXacAykwwLV3vNLKxPEhA17jxz7cpjfu3JZokLq1TH
 | 
			
		||||
 | 
			
		||||
Für die Organisation unserer Ausfahrten nutzen wir app.rudernlinz.at. Logge dich einfach mit deinem Namen ('{0}' ohne Anführungszeichen) ein, beim ersten Mal kannst du das Passwortfeld leer lassen. Unter 'Geplante Ausfahrten' kannst du dich jederzeit zu den Ausfahrten anmelden.
 | 
			
		||||
 | 
			
		||||
Beim nächsten Treffen im Verein, erinnere jemand vom Vorstand (https://rudernlinz.at/unser-verein/vorstand/) bitte daran, deinen Fingerabdruck zu registrieren, damit du Zugang zum Bootshaus erhältst.
 | 
			
		||||
 | 
			
		||||
Damit du dich noch mehr verbunden fühlst (:-)), haben wir im Bootshaus ein WLAN für Vereinsmitglieder 'ASKÖ Ruderverein Donau Linz' eingerichtet. Das Passwort dafür lautet 'donau1921' (ohne Anführungszeichen). Bitte gib das Passwort an keine vereinsfremden Personen weiter.
 | 
			
		||||
 | 
			
		||||
Wir freuen uns darauf, dich bald am Wasser zu sehen und gemeinsam tolle Erfahrungen zu sammeln!
 | 
			
		||||
 | 
			
		||||
Riemen- & Dollenbruch
 | 
			
		||||
ASKÖ Ruderverein Donau Linz", self.name),
 | 
			
		||||
            smtp_pw,
 | 
			
		||||
        ).await?;
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    async fn notify_coxes_about_new_regular(&self, db: &SqlitePool) {
 | 
			
		||||
        Notification::create_for_steering_people(
 | 
			
		||||
            db,
 | 
			
		||||
            &format!(
 | 
			
		||||
                "Liebe Steuerberechtigte, seit {} gibt es ein neues Mitglied: {}",
 | 
			
		||||
                self.member_since_date.clone().unwrap(),
 | 
			
		||||
                self.name
 | 
			
		||||
            ),
 | 
			
		||||
            "Neues Vereinsmitglied",
 | 
			
		||||
            None,
 | 
			
		||||
            None,
 | 
			
		||||
        )
 | 
			
		||||
        .await;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,8 +1,7 @@
 | 
			
		||||
use super::member::Member;
 | 
			
		||||
use super::regular::RegularUser;
 | 
			
		||||
use super::{ManageUserUser, User};
 | 
			
		||||
use crate::model::role::Role;
 | 
			
		||||
use crate::model::user::LoginError;
 | 
			
		||||
use crate::tera::admin::user::ScheckToRegularForm;
 | 
			
		||||
use crate::NonEmptyString;
 | 
			
		||||
use crate::{
 | 
			
		||||
    model::{mail::Mail, notification::Notification},
 | 
			
		||||
    special_user, SCHECKBUCH,
 | 
			
		||||
@@ -10,13 +9,7 @@ use crate::{
 | 
			
		||||
use chrono::NaiveDate;
 | 
			
		||||
use rocket::async_trait;
 | 
			
		||||
use rocket::fs::TempFile;
 | 
			
		||||
use rocket::http::Status;
 | 
			
		||||
use rocket::request;
 | 
			
		||||
use rocket::request::FromRequest;
 | 
			
		||||
use rocket::request::Outcome;
 | 
			
		||||
use rocket::Request;
 | 
			
		||||
use sqlx::SqlitePool;
 | 
			
		||||
use std::ops::Deref;
 | 
			
		||||
 | 
			
		||||
special_user!(ScheckbuchUser, +"scheckbuch");
 | 
			
		||||
 | 
			
		||||
@@ -24,11 +17,12 @@ impl ScheckbuchUser {
 | 
			
		||||
    pub(crate) async fn convert_to_regular_user(
 | 
			
		||||
        self,
 | 
			
		||||
        db: &SqlitePool,
 | 
			
		||||
        smtp_pw: &str,
 | 
			
		||||
        changed_by: &ManageUserUser,
 | 
			
		||||
        member_since: &NaiveDate,
 | 
			
		||||
        birthdate: &NaiveDate,
 | 
			
		||||
        phone: &str,
 | 
			
		||||
        address: &str,
 | 
			
		||||
        phone: NonEmptyString,
 | 
			
		||||
        address: NonEmptyString,
 | 
			
		||||
        membership_pdf: &TempFile<'_>,
 | 
			
		||||
    ) -> Result<(), String> {
 | 
			
		||||
        // Set data
 | 
			
		||||
@@ -36,9 +30,9 @@ impl ScheckbuchUser {
 | 
			
		||||
        self.user
 | 
			
		||||
            .update_member_since(db, changed_by, member_since)
 | 
			
		||||
            .await;
 | 
			
		||||
        self.user.update_phone(db, changed_by, phone).await?;
 | 
			
		||||
        self.user.update_address(db, changed_by, address).await?;
 | 
			
		||||
        self.user.update_address(db, changed_by, address).await?;
 | 
			
		||||
 | 
			
		||||
        self.user.update_phone(db, changed_by, &phone).await;
 | 
			
		||||
        self.user.update_address(db, changed_by, &address).await;
 | 
			
		||||
        self.user
 | 
			
		||||
            .add_membership_pdf(db, changed_by, membership_pdf)
 | 
			
		||||
            .await?;
 | 
			
		||||
@@ -50,25 +44,13 @@ impl ScheckbuchUser {
 | 
			
		||||
        self.user.add_role(db, changed_by, ®ular).await?;
 | 
			
		||||
 | 
			
		||||
        // Notify
 | 
			
		||||
        todo!() // Continue here
 | 
			
		||||
        let regular = RegularUser::new(db, &self.user).await.unwrap();
 | 
			
		||||
        regular.notify(db, smtp_pw).await?;
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    //async fn from(user: User, db: &SqlitePool, mail: &str, smtp_pw: &str) -> Result<(), String> {
 | 
			
		||||
    //    if user.has_role(db, "scheckbuch").await {
 | 
			
		||||
    //        return Err("User is already a scheckbuch".into());
 | 
			
		||||
    //    }
 | 
			
		||||
 | 
			
		||||
    //    // TODO: do we allow e.g. DonauLinz to scheckbuch?
 | 
			
		||||
 | 
			
		||||
    //    let scheckbuch = Role::find_by_name(db, "scheckbuch").await.unwrap();
 | 
			
		||||
    //    user.add_role(db, &scheckbuch).await.unwrap();
 | 
			
		||||
 | 
			
		||||
    //    // TODO: remove all other `membership_type` roles
 | 
			
		||||
    //    let new_user = Self::new(db, &user).await.unwrap();
 | 
			
		||||
 | 
			
		||||
    //    new_user.notify(db, mail, smtp_pw).await
 | 
			
		||||
    //}
 | 
			
		||||
 | 
			
		||||
    // TODO: make private
 | 
			
		||||
    pub(crate) async fn notify(
 | 
			
		||||
        &self,
 | 
			
		||||
        db: &SqlitePool,
 | 
			
		||||
 
 | 
			
		||||
@@ -390,13 +390,11 @@ async fn update_phone(
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    match user.update_phone(db, &admin, &data.phone).await {
 | 
			
		||||
        Ok(_) => Flash::success(
 | 
			
		||||
            Redirect::to(format!("/admin/user/{}", user.id)),
 | 
			
		||||
            "Telefonnummer erfolgreich geändert",
 | 
			
		||||
        ),
 | 
			
		||||
        Err(e) => Flash::error(Redirect::to(format!("/admin/user/{}", user.id)), e),
 | 
			
		||||
    }
 | 
			
		||||
    user.update_phone(db, &admin, &data.phone).await;
 | 
			
		||||
    Flash::success(
 | 
			
		||||
        Redirect::to(format!("/admin/user/{}", user.id)),
 | 
			
		||||
        "Telefonnummer erfolgreich geändert",
 | 
			
		||||
    )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(FromForm, Debug)]
 | 
			
		||||
@@ -418,13 +416,12 @@ async fn update_address(
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    match user.update_address(db, &admin, &data.address).await {
 | 
			
		||||
        Ok(_) => Flash::success(
 | 
			
		||||
            Redirect::to(format!("/admin/user/{}", user.id)),
 | 
			
		||||
            "Adresse erfolgreich geändert",
 | 
			
		||||
        ),
 | 
			
		||||
        Err(e) => Flash::error(Redirect::to(format!("/admin/user/{}", user.id)), e),
 | 
			
		||||
    }
 | 
			
		||||
    user.update_address(db, &admin, &data.address).await;
 | 
			
		||||
 | 
			
		||||
    Flash::success(
 | 
			
		||||
        Redirect::to(format!("/admin/user/{}", user.id)),
 | 
			
		||||
        "Adresse erfolgreich geändert",
 | 
			
		||||
    )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(FromForm, Debug)]
 | 
			
		||||
@@ -831,6 +828,7 @@ async fn scheckbook_to_regular(
 | 
			
		||||
    db: &State<SqlitePool>,
 | 
			
		||||
    data: Form<ScheckToRegularForm<'_>>,
 | 
			
		||||
    admin: ManageUserUser,
 | 
			
		||||
    config: &State<Config>,
 | 
			
		||||
    id: i32,
 | 
			
		||||
) -> Flash<Redirect> {
 | 
			
		||||
    let Some(user) = User::find_by_id(db, id).await else {
 | 
			
		||||
@@ -842,13 +840,19 @@ async fn scheckbook_to_regular(
 | 
			
		||||
    let Ok(birthdate) = NaiveDate::parse_from_str(&data.birthdate, "%Y-%m-%d") else {
 | 
			
		||||
        return Flash::error(
 | 
			
		||||
            Redirect::to(format!("/admin/user/{id}")),
 | 
			
		||||
            format!("Datum {} ist nicht im YYYY-MM-DD Format", &data.birthdate),
 | 
			
		||||
            format!(
 | 
			
		||||
                "Geburtsdatum {} ist nicht im YYYY-MM-DD Format",
 | 
			
		||||
                &data.birthdate
 | 
			
		||||
            ),
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
    let Ok(member_since) = NaiveDate::parse_from_str(&data.member_since, "%Y-%m-%d") else {
 | 
			
		||||
        return Flash::error(
 | 
			
		||||
            Redirect::to(format!("/admin/user/{id}")),
 | 
			
		||||
            format!("Datum {} ist nicht im YYYY-MM-DD Format", &data.birthdate),
 | 
			
		||||
            format!(
 | 
			
		||||
                "Beitrittsdatum {} ist nicht im YYYY-MM-DD Format",
 | 
			
		||||
                &data.birthdate
 | 
			
		||||
            ),
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
@@ -859,14 +863,28 @@ async fn scheckbook_to_regular(
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    let Ok(phone) = data.phone.clone().try_into() else {
 | 
			
		||||
        return Flash::error(
 | 
			
		||||
            Redirect::to(format!("/admin/user/{id}")),
 | 
			
		||||
            "Vereinsmitglied braucht eine Telefonnummer",
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
    let Ok(address) = data.address.clone().try_into() else {
 | 
			
		||||
        return Flash::error(
 | 
			
		||||
            Redirect::to(format!("/admin/user/{id}")),
 | 
			
		||||
            "Vereinsmitglied braucht eine Adresse",
 | 
			
		||||
        );
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    match user
 | 
			
		||||
        .convert_to_regular_user(
 | 
			
		||||
            db,
 | 
			
		||||
            &config.smtp_pw,
 | 
			
		||||
            &admin,
 | 
			
		||||
            &member_since,
 | 
			
		||||
            &birthdate,
 | 
			
		||||
            &data.phone,
 | 
			
		||||
            &data.address,
 | 
			
		||||
            phone,
 | 
			
		||||
            address,
 | 
			
		||||
            &data.membership_pdf,
 | 
			
		||||
        )
 | 
			
		||||
        .await
 | 
			
		||||
 
 | 
			
		||||
@@ -4,323 +4,329 @@
 | 
			
		||||
{% block content %}
 | 
			
		||||
    <div class="max-w-screen-lg w-full">
 | 
			
		||||
        <h1 class="h1">{{ user.name }}</h1>
 | 
			
		||||
        <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
             role="alert">
 | 
			
		||||
            <h2 class="h2">Grunddaten</h2>
 | 
			
		||||
            <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                <div class="py-3">
 | 
			
		||||
                    {% if user.last_access %}
 | 
			
		||||
                        Zuletzt eingeloggt am {{ user.last_access | date(format="%d. %m. %Y") }}
 | 
			
		||||
                    {% else %}
 | 
			
		||||
                        {{ user.name }} hat sich noch nie eingeloggt.
 | 
			
		||||
        <div class="grid sm:grid-cols-2 gap-3">
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">
 | 
			
		||||
                    Grunddaten
 | 
			
		||||
                    <br />
 | 
			
		||||
                    <small class="inline-block text-xs  text-gray-500 dark:text-gray-100 ">
 | 
			
		||||
                        {% if user.last_access %}
 | 
			
		||||
                            Zuletzt eingeloggt am {{ user.last_access | date(format="%d. %m. %Y") }}
 | 
			
		||||
                        {% else %}
 | 
			
		||||
                            {{ user.name }} hat sich noch nie eingeloggt.
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                    </small>
 | 
			
		||||
                </h2>
 | 
			
		||||
                <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                    <div class="py-3 grid gap-3">
 | 
			
		||||
                        <form action="/admin/user/{{ user.id }}/change-mail" method="post">
 | 
			
		||||
                            {{ macros::inputgroup(label='Mailadresse', name='mail', type="text", value=user.mail, readonly=not allowed_to_edit) }}
 | 
			
		||||
                        </form>
 | 
			
		||||
                        <form action="/admin/user/{{ user.id }}/change-phone" method="post">
 | 
			
		||||
                            {{ macros::inputgroup(label='Telefonnummer', name='phone', type="text", value=user.phone, readonly=not allowed_to_edit) }}
 | 
			
		||||
                        </form>
 | 
			
		||||
                        <form action="/admin/user/{{ user.id }}/change-nickname" method="post">
 | 
			
		||||
                            {{ macros::inputgroup(label='Spitzname', name='nickname', type="text", value=user.nickname, readonly=not allowed_to_edit) }}
 | 
			
		||||
                        </form>
 | 
			
		||||
                        <span>Notizen: to be replaced with activity :-)</span>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">
 | 
			
		||||
                    Mitgliedschaft
 | 
			
		||||
                    <br />
 | 
			
		||||
                    <small class="inline-block text-xs  text-gray-500 dark:text-gray-100 ">
 | 
			
		||||
                        {% if "SchnupperInterest" in member %}
 | 
			
		||||
                            Interessiert am Schnupperkurs
 | 
			
		||||
                        {% elif "Schnupperant" in member %}
 | 
			
		||||
                            Beim nächsten Schnupperkurs angemeldet
 | 
			
		||||
                        {% elif "Scheckbuch" in member %}
 | 
			
		||||
                            {% set logbook = member["Scheckbuch"] %}
 | 
			
		||||
                            Scheckbuch (Ausfahrten: {{ logbook | length }})
 | 
			
		||||
                        {% elif "Regular" in member %}
 | 
			
		||||
                            Reguläres Vereinsmitglied
 | 
			
		||||
                        {% elif "Foerdernd" in member %}
 | 
			
		||||
                            Förderndes Vereinsmitglied
 | 
			
		||||
                        {% elif "Unterstuetzend" in member %}
 | 
			
		||||
                            Unterstützendes Vereinsmitglied
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                    </small>
 | 
			
		||||
                </h2>
 | 
			
		||||
                <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                    {% if is_clubmember %}
 | 
			
		||||
                        <div class="py-3 grid gap-3">
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-member-since" method="post">
 | 
			
		||||
                                {{ macros::inputgroup(label='Mitglied seit', name='member_since', type="date", value=user.member_since_date, readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-birthdate" method="post">
 | 
			
		||||
                                {{ macros::inputgroup(label='Geburtsdatum', name='birthdate', type="date", value=user.birthdate, readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-address" method="post">
 | 
			
		||||
                                {{ macros::inputgroup(label='Adresse', name='address', type="text", value=user.address, readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-family" method="post">
 | 
			
		||||
                                {{ macros::selectgroup(label="Familie", data=families, name='family_id', selected_id=user.family_id, display=['names'], default="Keine Familie", new_last_entry='Neue Familie anlegen', readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                        </div>
 | 
			
		||||
                        <div class="py-3">
 | 
			
		||||
                            {% if user.membership_pdf %}
 | 
			
		||||
                                <a href="/admin/user/{{ user.id }}/membership" class="link link-primary">Beitrittserklärung herunterladen</a>
 | 
			
		||||
                            {% else %}
 | 
			
		||||
                                ⚠️ Aktuell gibt's keine Beitrittserklärung 😢
 | 
			
		||||
                                {% if allowed_to_edit %}
 | 
			
		||||
                                    Das kannst du hier ändern ⤵️
 | 
			
		||||
                                    <form action="/admin/user/{{ user.id }}/add-membership-pdf"
 | 
			
		||||
                                          method="post"
 | 
			
		||||
                                          enctype="multipart/form-data"
 | 
			
		||||
                                          class="grid gap-3">
 | 
			
		||||
                                        <fieldset>
 | 
			
		||||
                                            {{ macros::input(label='Neue Beitrittserklärung hochladen', name='membership_pdf', type="file", accept='application/pdf') }}
 | 
			
		||||
                                        </fieldset>
 | 
			
		||||
                                        <input value="Hochladen" type="submit" class="btn btn-primary" />
 | 
			
		||||
                                    </form>
 | 
			
		||||
                                {% endif %}
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        </div>
 | 
			
		||||
                        {% if allowed_to_edit %}
 | 
			
		||||
                            <div class="py-3">
 | 
			
		||||
                                <div class="mt-3 text-right">
 | 
			
		||||
                                    <a href="/admin/user/{{ user.id }}/delete"
 | 
			
		||||
                                       class="btn btn-alert"
 | 
			
		||||
                                       onclick="return confirm('Ist {{ user.name }} wirklich aus dem Verein ausgetreten?');">
 | 
			
		||||
                                        {% include "includes/delete-icon" %}
 | 
			
		||||
                                        Mitglied ist ausgetreten
 | 
			
		||||
                                    </a>
 | 
			
		||||
                                </div>
 | 
			
		||||
                            </div>
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                    {% elif "Scheckbuch" in member %}
 | 
			
		||||
                        <div class="grid gap-3 pb-3">
 | 
			
		||||
                            {% for log in logbook %}
 | 
			
		||||
                                {{ log::show_old(log=log, state="completed", only_ones=false, index=loop.index, allowed_to_edit=false) }}
 | 
			
		||||
                            {% endfor %}
 | 
			
		||||
                            <button type="button"
 | 
			
		||||
                                    onclick="document.getElementById('call-for-action').showModal()"
 | 
			
		||||
                                    class="btn btn-primary">Zu Vereinsmitglied umwandeln</button>
 | 
			
		||||
                        </div>
 | 
			
		||||
                        <dialog id="call-for-action"
 | 
			
		||||
                                class="max-w-screen-sm w-full dark:bg-primary-600 dark:text-white rounded-md"
 | 
			
		||||
                                onclick="document.getElementById('call-for-action').close()">
 | 
			
		||||
                            <div onclick="event.stopPropagation();" class="p-3">
 | 
			
		||||
                                <button type="button"
 | 
			
		||||
                                        onclick="document.getElementById('call-for-action').close()"
 | 
			
		||||
                                        title="Schließen"
 | 
			
		||||
                                        class="sidebar-close border-0 bg-primary-100 focus:bg-primary-50 text-black flex items-center justify-center transform rotate-45 absolute right-0 mr-3">
 | 
			
		||||
                                    <svg class="inline h-5 w-5"
 | 
			
		||||
                                         width="16"
 | 
			
		||||
                                         height="16"
 | 
			
		||||
                                         fill="currentColor"
 | 
			
		||||
                                         viewBox="0 0 16 16">
 | 
			
		||||
                                        <path d="M8 4a.5.5 0 0 1 .5.5v3h3a.5.5 0 0 1 0 1h-3v3a.5.5 0 0 1-1 0v-3h-3a.5.5 0 0 1 0-1h3v-3A.5.5 0 0 1 8 4z"></path>
 | 
			
		||||
                                    </svg>
 | 
			
		||||
                                </button>
 | 
			
		||||
                                <div class="mt-8">
 | 
			
		||||
                                    <form action="/admin/user/{{ user.id }}/scheckbook-to-regular"
 | 
			
		||||
                                          method="post"
 | 
			
		||||
                                          enctype="multipart/form-data"
 | 
			
		||||
                                          class="grid gap-3">
 | 
			
		||||
                                        Type: Select -> normales Mitglied, förderndes Mitglied, unterstützendes Mitglied
 | 
			
		||||
                                        {{ macros::input(label='Mitglied seit', name='member_since', type="date", value=now() | date(), required=true) }}
 | 
			
		||||
                                        {{ macros::input(label='Geburtsdatum', name='birthdate', type="date", value=user.birthdate, required=true) }}
 | 
			
		||||
                                        {{ macros::input(label='Telefonnummer', name='phone', type="text", value=user.phone, required=true) }}
 | 
			
		||||
                                        {{ macros::input(label='Adresse', name='address', type="text", value=user.address, required=true) }}
 | 
			
		||||
                                        {{ macros::input(label='Beitrittserklärung', name='membership_pdf', type="file", accept='application/pdf', required=true) }}
 | 
			
		||||
                                        <input value="Als neues, reguläres Mitglied anlegen"
 | 
			
		||||
                                               type="submit"
 | 
			
		||||
                                               class="btn btn-primary" />
 | 
			
		||||
                                    </form>
 | 
			
		||||
                                </div>
 | 
			
		||||
                            </div>
 | 
			
		||||
                        </dialog>
 | 
			
		||||
                    {% endif %}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="py-3">
 | 
			
		||||
                    <ul class="grid gap-3">
 | 
			
		||||
                        <li>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-mail" method="post">
 | 
			
		||||
                                {{ macros::inputgroup(label='Mailadresse', name='mail', type="text", value=user.mail, readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                        </li>
 | 
			
		||||
                        <li>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-phone" method="post">
 | 
			
		||||
                                {{ macros::inputgroup(label='Telefonnummer', name='phone', type="text", value=user.phone, readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                        </li>
 | 
			
		||||
                        <li>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/change-nickname" method="post">
 | 
			
		||||
                                {{ macros::inputgroup(label='Spitzname', name='nickname', type="text", value=user.nickname, readonly=not allowed_to_edit) }}
 | 
			
		||||
                            </form>
 | 
			
		||||
                        </li>
 | 
			
		||||
                        <li>Notizen: to be replaced with activity :-)</li>
 | 
			
		||||
                    </ul>
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="py-3">
 | 
			
		||||
                    Rollen:
 | 
			
		||||
                    <ul class="list-disc ms-4">
 | 
			
		||||
                        {% for role in user.proper_roles -%}
 | 
			
		||||
                            {% if not role.cluster and not role.hide_in_lists %}
 | 
			
		||||
                                <li>
 | 
			
		||||
                                    <strong>
 | 
			
		||||
                                        {% if role.formatted_name %}
 | 
			
		||||
                                            {{ role.formatted_name }}
 | 
			
		||||
                                        {% else %}
 | 
			
		||||
                                            {{ role.name }}
 | 
			
		||||
                                        {% endif %}
 | 
			
		||||
                                    </strong>  {{ role.desc }}
 | 
			
		||||
                                    {% if allowed_to_edit %}
 | 
			
		||||
                                        <a href="/admin/user/{{ user.id }}/remove-role/{{ role.id }}"
 | 
			
		||||
                                           onclick="return confirm('Willst du die Rolle \'{{ role.name }}\' von {{ user.name }} wirklich entfernen?');">🗑️</a>
 | 
			
		||||
                                    {% endif %}
 | 
			
		||||
                                </li>
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        {% endfor %}
 | 
			
		||||
                    </ul>
 | 
			
		||||
                    {% if allowed_to_edit %}
 | 
			
		||||
                        <details>
 | 
			
		||||
                            <summary>+ Rolle</summary>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/add-role" method="post">
 | 
			
		||||
                                <fieldset>
 | 
			
		||||
                                    <select name="role_id">
 | 
			
		||||
                                        {% for role in roles %}
 | 
			
		||||
                                            {% if not role.cluster and role not in user.proper_roles and not role.hide_in_lists %}
 | 
			
		||||
                                                <option value="{{ role.id }}">
 | 
			
		||||
            </div>
 | 
			
		||||
            {% if is_clubmember %}
 | 
			
		||||
                <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                     role="alert">
 | 
			
		||||
                    <h2 class="h2">Rollen</h2>
 | 
			
		||||
                    <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                        <div class="py-3">
 | 
			
		||||
                            <ul>
 | 
			
		||||
                                {% for role in user.proper_roles -%}
 | 
			
		||||
                                    {% if not role.cluster and not role.hide_in_lists %}
 | 
			
		||||
                                        <li class="flex my-2 w-full justify-between items-center hover:bg-gray-100">
 | 
			
		||||
                                            <span>
 | 
			
		||||
                                                <strong>
 | 
			
		||||
                                                    {% if role.formatted_name %}
 | 
			
		||||
                                                        {{ role.formatted_name }}
 | 
			
		||||
                                                    {% else %}
 | 
			
		||||
                                                        {{ role.name }}
 | 
			
		||||
                                                    {% endif %}
 | 
			
		||||
                                                    {% if role.desc %}({{ role.desc }}){% endif %}
 | 
			
		||||
                                                </option>
 | 
			
		||||
                                                </strong>
 | 
			
		||||
                                                <br />
 | 
			
		||||
                                                <small>{{ role.desc }}</small>
 | 
			
		||||
                                            </span>
 | 
			
		||||
                                            {% if allowed_to_edit %}
 | 
			
		||||
                                                <a href="/admin/user/{{ user.id }}/remove-role/{{ role.id }}"
 | 
			
		||||
                                                   onclick="return confirm('Willst du die Rolle \'{{ role.name }}\' von {{ user.name }} wirklich entfernen?');">🗑️</a>
 | 
			
		||||
                                            {% endif %}
 | 
			
		||||
                                        {% endfor %}
 | 
			
		||||
                                    </select>
 | 
			
		||||
                                    <input value="Rolle hinzufügen" type="submit" class="btn btn-primary ml-1" />
 | 
			
		||||
                                </fieldset>
 | 
			
		||||
                            </form>
 | 
			
		||||
                        </details>
 | 
			
		||||
                    {% endif %}
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        {% if supposed_to_pay %}
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">💸</h2>
 | 
			
		||||
                <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                    <div class="py-3">
 | 
			
		||||
                        {% if fee %}
 | 
			
		||||
                            <div>
 | 
			
		||||
                                <strong>{{ fee.name }}</strong>
 | 
			
		||||
                                <span class="block">{{ fee.sum_in_cents / 100 }}€</span>
 | 
			
		||||
                            </div>
 | 
			
		||||
                            <div>
 | 
			
		||||
                                {% for p in fee.parts %}
 | 
			
		||||
                                    {{ p.0 }} ({{ p.1 / 100 }}€)
 | 
			
		||||
                                    {% if not loop.last %}+{% endif %}
 | 
			
		||||
                                {% endfor %}
 | 
			
		||||
                            </div>
 | 
			
		||||
                            {% if "paid" in user.roles %}
 | 
			
		||||
                                ✅ bezahlt
 | 
			
		||||
                            {% else %}
 | 
			
		||||
                                ❌ Zahlung ausständig
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        {% else %}
 | 
			
		||||
                            {% if "paid" in user.roles %}
 | 
			
		||||
                                ✅ {{ member | keys }} hat schon bezahlt
 | 
			
		||||
                            {% else %}
 | 
			
		||||
                                ❌
 | 
			
		||||
                                {% for key, value in member %}
 | 
			
		||||
                                    {% if loop.first %}{{ key }}{% endif %}
 | 
			
		||||
                                {% endfor %}
 | 
			
		||||
                                hat noch nicht bezahlt
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        {% endif %}
 | 
			
		||||
        {% if is_clubmember %}
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">Vereinsmitglied</h2>
 | 
			
		||||
                <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                    <div class="py-3">
 | 
			
		||||
                        <ul class="grid gap-3">
 | 
			
		||||
                            <li>
 | 
			
		||||
                                <form action="/admin/user/{{ user.id }}/change-member-since" method="post">
 | 
			
		||||
                                    {{ macros::inputgroup(label='Mitglied seit', name='member_since', type="date", value=user.member_since_date, readonly=not allowed_to_edit) }}
 | 
			
		||||
                                </form>
 | 
			
		||||
                            </li>
 | 
			
		||||
                            <li>
 | 
			
		||||
                                <form action="/admin/user/{{ user.id }}/change-birthdate" method="post">
 | 
			
		||||
                                    {{ macros::inputgroup(label='Geburtsdatum', name='birthdate', type="date", value=user.birthdate, readonly=not allowed_to_edit) }}
 | 
			
		||||
                                </form>
 | 
			
		||||
                            </li>
 | 
			
		||||
                            <li>
 | 
			
		||||
                                <form action="/admin/user/{{ user.id }}/change-address" method="post">
 | 
			
		||||
                                    {{ macros::inputgroup(label='Adresse', name='address', type="text", value=user.address, readonly=not allowed_to_edit) }}
 | 
			
		||||
                                </form>
 | 
			
		||||
                            </li>
 | 
			
		||||
                             <li>
 | 
			
		||||
                                Familie:
 | 
			
		||||
                                {% for family in families %}
 | 
			
		||||
                                    {% if user.family_id == family.id %}{{ family.names }}{% endif %}
 | 
			
		||||
                                {% endfor %}
 | 
			
		||||
                                {% if allowed_to_edit %}
 | 
			
		||||
                                    <details>
 | 
			
		||||
                                        <summary>✏️</summary>
 | 
			
		||||
                                        <form action="/admin/user/{{ user.id }}/change-family" method="post">
 | 
			
		||||
                                            {{ macros::select(label="Familie", data=families, name='family_id', selected_id=user.family_id, display=['names'], default="Keine Familie", new_last_entry='Neue Familie anlegen') }}
 | 
			
		||||
                                            <input value="Ändern" type="submit" class="btn btn-primary ml-1" />
 | 
			
		||||
                                        </form>
 | 
			
		||||
                                    </details>
 | 
			
		||||
                                {% endif %}
 | 
			
		||||
                            </li>
 | 
			
		||||
                        </ul>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <div class="py-3">
 | 
			
		||||
                        {% if user.membership_pdf %}
 | 
			
		||||
                            <a href="/admin/user/{{ user.id }}/membership"
 | 
			
		||||
                               class="text-black dark:text-white">Beitrittserklärung</a>
 | 
			
		||||
                        {% else %}
 | 
			
		||||
                            ⚠️ Aktuell gibt's keine Beitrittserklärung 😢
 | 
			
		||||
                            {% if allowed_to_edit %}
 | 
			
		||||
                                Das kannst du hier ändern ⤵️
 | 
			
		||||
                                <form action="/admin/user/{{ user.id }}/add-membership-pdf"
 | 
			
		||||
                                      method="post"
 | 
			
		||||
                                      enctype="multipart/form-data">
 | 
			
		||||
                                    <fieldset>
 | 
			
		||||
                                        {{ macros::input(label='Neue Beitrittserklärung hochladen', name='membership_pdf', type="file", accept='application/pdf') }}
 | 
			
		||||
                                        <input value="Hochladen" type="submit" class="btn btn-primary ml-1" />
 | 
			
		||||
                                    </fieldset>
 | 
			
		||||
                                </form>
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                    </div>
 | 
			
		||||
                    {% if allowed_to_edit %}
 | 
			
		||||
                        <div class="py-3">
 | 
			
		||||
                            <div class="mt-3 text-right">
 | 
			
		||||
                                <a href="/admin/user/{{ user.id }}/delete"
 | 
			
		||||
                                   class="btn btn-alert"
 | 
			
		||||
                                   onclick="return confirm('Ist {{ user.name }} wirklich aus dem Verein ausgetreten?');">
 | 
			
		||||
                                    {% include "includes/delete-icon" %}
 | 
			
		||||
                                    Mitglied ist ausgetreten
 | 
			
		||||
                                </a>
 | 
			
		||||
                            </div>
 | 
			
		||||
                        </div>
 | 
			
		||||
                    {% endif %}
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        {% endif %}
 | 
			
		||||
        <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
             role="alert">
 | 
			
		||||
            <h2 class="h2">Mitgliedstyp</h2>
 | 
			
		||||
            <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                <div class="py-3">
 | 
			
		||||
                    {{ user.name }}
 | 
			
		||||
                    {% if "SchnupperInterest" in member %}
 | 
			
		||||
                        ist interessiert am Schnupperkurs.
 | 
			
		||||
                    {% elif "Schnupperant" in member %}
 | 
			
		||||
                        ist beim nächsten Schnupperkurs angemeldet.
 | 
			
		||||
                    {% elif "Scheckbuch" in member %}
 | 
			
		||||
                        {% set logbook = member["Scheckbuch"] %}
 | 
			
		||||
                        hat ein Scheckbuch und {{ logbook | length }} Ausfahrten absolviert.
 | 
			
		||||
                        <details>
 | 
			
		||||
                            <summary>Ausfahrten</summary>
 | 
			
		||||
                            {% for log in logbook %}
 | 
			
		||||
                                {{ log::show_old(log=log, state="completed", only_ones=false, index=loop.index, allowed_to_edit=false) }}
 | 
			
		||||
                            {% endfor %}
 | 
			
		||||
                        </details>
 | 
			
		||||
                        <details>
 | 
			
		||||
                            <summary>Zu reguläres Vereinsmitglied umwandeln</summary>
 | 
			
		||||
                            <form action="/admin/user/{{ user.id }}/scheckbook-to-regular"
 | 
			
		||||
                                  method="post"
 | 
			
		||||
                                  enctype="multipart/form-data">
 | 
			
		||||
                                {{ macros::input(label='Mitglied seit', name='member_since', type="date", value=now() | date()) }}
 | 
			
		||||
                                {{ macros::input(label='Geburtsdatum', name='birthdate', type="date", value=user.birthdate) }}
 | 
			
		||||
                                {{ macros::input(label='Telefonnummer', name='phone', type="text", value=user.phone) }}
 | 
			
		||||
                                {{ macros::input(label='Adresse', name='address', type="text", value=user.address) }}
 | 
			
		||||
                                {{ macros::input(label='Beitrittserklärung', name='membership_pdf', type="file", accept='application/pdf') }}
 | 
			
		||||
                                <input value="Als neues, reguläres Mitglied anlegen"
 | 
			
		||||
                                       type="submit"
 | 
			
		||||
                                       class="btn btn-primary ml-1" />
 | 
			
		||||
                            </form>
 | 
			
		||||
                        </details>
 | 
			
		||||
                    {% elif "Regular" in member %}
 | 
			
		||||
                        ist ein reguläres Vereinsmitglied.
 | 
			
		||||
                    {% elif "Foerdernd" in member %}
 | 
			
		||||
                        ist ein förderndes Vereinsmitglied.
 | 
			
		||||
                    {% elif "Unterstuetzend" in member %}
 | 
			
		||||
                        ist ein unterstützendes Vereinsmitglied.
 | 
			
		||||
                    {% endif %}
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
             role="alert">
 | 
			
		||||
            <h2 class="h2">Aktivität von und mit {{ user.name }}</h2>
 | 
			
		||||
            <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                <div class="py-3">
 | 
			
		||||
                    <ul class="list-disc ms-4">
 | 
			
		||||
                        <li>Passwort zurückgesetzt am/um X</li>
 | 
			
		||||
                        <li>Am X beigetreten.</li>
 | 
			
		||||
                    </ul>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
             role="alert">
 | 
			
		||||
            <h2 class="h2">TODO</h2>
 | 
			
		||||
            <div class="border-t bg-white dark:bg-primary-900 py-3 px-4 relative">
 | 
			
		||||
                <span class="text-black dark:text-white cursor-pointer">
 | 
			
		||||
                    <span class="font-bold">
 | 
			
		||||
                        {{ user.name }}
 | 
			
		||||
                        {% if not user.last_access and allowed_to_edit and user.mail %}
 | 
			
		||||
                            <form action="/admin/user"
 | 
			
		||||
                                  method="post"
 | 
			
		||||
                                  enctype="multipart/form-data"
 | 
			
		||||
                                  class="inline">
 | 
			
		||||
                                • <a class="font-normal text-primary-600 dark:text-primary-200 hover:text-primary-900 dark:hover:text-primary-300 underline"
 | 
			
		||||
    href="/admin/user/{{ user.id }}/send-welcome-mail"
 | 
			
		||||
    onclick="return confirm('Willst du wirklich das Willkommensmail an {{ user.name }} ausschicken?');">Willkommensmail verschicken</a>
 | 
			
		||||
                            </form>
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                    </span>
 | 
			
		||||
                </span>
 | 
			
		||||
                <form action="/admin/user"
 | 
			
		||||
                      method="post"
 | 
			
		||||
                      enctype="multipart/form-data"
 | 
			
		||||
                      class="w-full mt-2">
 | 
			
		||||
                    {% if user.pw %}
 | 
			
		||||
                        <a class="block my-1 font-normal text-[#f43f5e] dark:text-primary-200 hover:text-primary-900 dark:hover:text-primary-300 underline"
 | 
			
		||||
                           href="/admin/user/{{ user.id }}/reset-pw"
 | 
			
		||||
                           onclick="return confirm('Willst du wirklich das Passwort zurücksetzen?');">Passwort zurücksetzen</a>
 | 
			
		||||
                    {% endif %}
 | 
			
		||||
                    <div class="w-full grid gap-3 mt-3">
 | 
			
		||||
                        <input type="hidden" name="id" value="{{ user.id }}" />
 | 
			
		||||
                        <div class="grid sm:grid-cols-2 lg:grid-cols-4 gap-3">
 | 
			
		||||
                            {% for cluster, cluster_roles in roles | group_by(attribute="cluster") %}
 | 
			
		||||
                                <label for="cluster_{{ loop.index }}">{{ cluster }}</label>
 | 
			
		||||
                                {# Determine the initially selected role within the cluster #}
 | 
			
		||||
                                {% set_global selected_role_id = "none" %}
 | 
			
		||||
                                {% for role in cluster_roles %}
 | 
			
		||||
                                    {% if selected_role_id == "none" and role.name in user.roles %}
 | 
			
		||||
                                        {% set_global selected_role_id = role.id %}
 | 
			
		||||
                                        </li>
 | 
			
		||||
                                    {% endif %}
 | 
			
		||||
                                {% endfor %}
 | 
			
		||||
                                {# Set default name to the selected role ID or first role if none selected #}
 | 
			
		||||
                                <select id="cluster_{{ loop.index }}"
 | 
			
		||||
                                        {% if selected_role_id == 'none' %} {% else %} name="roles[{{ selected_role_id }}]" {% endif %}
 | 
			
		||||
                                        {% if allowed_to_edit == false %}disabled{% endif %}
 | 
			
		||||
                                        onchange=" if (this.value === '') { this.removeAttribute('name'); } else { this.name = 'roles[' + this.options[this.selectedIndex].getAttribute('data-role-id') + ']'; }">
 | 
			
		||||
                                    <option value=""
 | 
			
		||||
                                            data-role-id="none"
 | 
			
		||||
                                            {% if selected_role_id == 'none' %}selected="selected"{% endif %}>
 | 
			
		||||
                                        None
 | 
			
		||||
                                    </option>
 | 
			
		||||
                                    {% for role in cluster_roles %}
 | 
			
		||||
                                        <option value="on"
 | 
			
		||||
                                                data-role-id="{{ role.id }}"
 | 
			
		||||
                                                {% if role.id == selected_role_id %}selected="selected"{% endif %}>
 | 
			
		||||
                                            {{ role.name }}
 | 
			
		||||
                                        </option>
 | 
			
		||||
                                    {% endfor %}
 | 
			
		||||
                                </select>
 | 
			
		||||
                            {% endfor %}
 | 
			
		||||
                            </ul>
 | 
			
		||||
                            {% if allowed_to_edit %}
 | 
			
		||||
                                <details>
 | 
			
		||||
                                    <summary>+ Rolle</summary>
 | 
			
		||||
                                    <form action="/admin/user/{{ user.id }}/add-role" method="post">
 | 
			
		||||
                                        <fieldset>
 | 
			
		||||
                                            <select name="role_id">
 | 
			
		||||
                                                {% for role in roles %}
 | 
			
		||||
                                                    {% if not role.cluster and role not in user.proper_roles and not role.hide_in_lists %}
 | 
			
		||||
                                                        <option value="{{ role.id }}">
 | 
			
		||||
                                                            {% if role.formatted_name %}
 | 
			
		||||
                                                                {{ role.formatted_name }}
 | 
			
		||||
                                                            {% else %}
 | 
			
		||||
                                                                {{ role.name }}
 | 
			
		||||
                                                            {% endif %}
 | 
			
		||||
                                                            {% if role.desc %}({{ role.desc }}){% endif %}
 | 
			
		||||
                                                        </option>
 | 
			
		||||
                                                    {% endif %}
 | 
			
		||||
                                                {% endfor %}
 | 
			
		||||
                                            </select>
 | 
			
		||||
                                            <input value="Rolle hinzufügen" type="submit" class="btn btn-primary ml-1" />
 | 
			
		||||
                                        </fieldset>
 | 
			
		||||
                                    </form>
 | 
			
		||||
                                </details>
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </form>
 | 
			
		||||
                </div>
 | 
			
		||||
            {% endif %}
 | 
			
		||||
            {% if supposed_to_pay %}
 | 
			
		||||
                <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                     role="alert">
 | 
			
		||||
                    <h2 class="h2">💸-Beitrag</h2>
 | 
			
		||||
                    <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                        <div class="py-3">
 | 
			
		||||
                            {% if fee %}
 | 
			
		||||
                                <div>
 | 
			
		||||
                                    <strong>{{ fee.name }}</strong>
 | 
			
		||||
                                    <span class="block">{{ fee.sum_in_cents / 100 }}€</span>
 | 
			
		||||
                                </div>
 | 
			
		||||
                                <div>
 | 
			
		||||
                                    {% for p in fee.parts %}
 | 
			
		||||
                                        {{ p.0 }} ({{ p.1 / 100 }}€)
 | 
			
		||||
                                        {% if not loop.last %}+{% endif %}
 | 
			
		||||
                                    {% endfor %}
 | 
			
		||||
                                </div>
 | 
			
		||||
                                {% if "paid" in user.roles %}
 | 
			
		||||
                                    ✅ bezahlt
 | 
			
		||||
                                {% else %}
 | 
			
		||||
                                    ❌ Zahlung ausständig
 | 
			
		||||
                                {% endif %}
 | 
			
		||||
                            {% else %}
 | 
			
		||||
                                {% if "paid" in user.roles %}
 | 
			
		||||
                                    ✅ {{ member | keys }} hat schon bezahlt
 | 
			
		||||
                                {% else %}
 | 
			
		||||
                                    ❌
 | 
			
		||||
                                    {% for key, value in member %}
 | 
			
		||||
                                        {% if loop.first %}{{ key }}{% endif %}
 | 
			
		||||
                                    {% endfor %}
 | 
			
		||||
                                    hat noch nicht bezahlt
 | 
			
		||||
                                {% endif %}
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            {% endif %}
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">Aktivität von und mit {{ user.name }}</h2>
 | 
			
		||||
                <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                    <div class="py-3">
 | 
			
		||||
                        <ul class="list-disc ms-4">
 | 
			
		||||
                            <li>Passwort zurückgesetzt am/um X</li>
 | 
			
		||||
                            <li>Am X beigetreten.</li>
 | 
			
		||||
                        </ul>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
             role="alert">
 | 
			
		||||
            <h2 class="h2">Ergo-Challenge</h2>
 | 
			
		||||
            <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                <div class="py-3">
 | 
			
		||||
                    {{ macros::input(label='DOB', name='dob', type="text", value=user.dob, readonly=allowed_to_edit == false) }}
 | 
			
		||||
                    {{ macros::input(label='Weight (kg)', name='weight', type="text", value=user.weight, readonly=allowed_to_edit == false) }}
 | 
			
		||||
                    {{ macros::input(label='Sex', name='sex', type="text", value=user.sex, readonly=allowed_to_edit == false) }}
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">TODO</h2>
 | 
			
		||||
                <div class="border-t bg-white dark:bg-primary-900 py-3 px-4 relative">
 | 
			
		||||
                    <span class="text-black dark:text-white cursor-pointer">
 | 
			
		||||
                        <span class="font-bold">
 | 
			
		||||
                            {{ user.name }}
 | 
			
		||||
                            {% if not user.last_access and allowed_to_edit and user.mail %}
 | 
			
		||||
                                <form action="/admin/user"
 | 
			
		||||
                                      method="post"
 | 
			
		||||
                                      enctype="multipart/form-data"
 | 
			
		||||
                                      class="inline">
 | 
			
		||||
                                    • <a class="font-normal text-primary-600 dark:text-primary-200 hover:text-primary-900 dark:hover:text-primary-300 underline"
 | 
			
		||||
    href="/admin/user/{{ user.id }}/send-welcome-mail"
 | 
			
		||||
    onclick="return confirm('Willst du wirklich das Willkommensmail an {{ user.name }} ausschicken?');">Willkommensmail verschicken</a>
 | 
			
		||||
                                </form>
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                        </span>
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <form action="/admin/user"
 | 
			
		||||
                          method="post"
 | 
			
		||||
                          enctype="multipart/form-data"
 | 
			
		||||
                          class="w-full mt-2">
 | 
			
		||||
                        {% if user.pw %}
 | 
			
		||||
                            <a class="block my-1 font-normal text-[#f43f5e] dark:text-primary-200 hover:text-primary-900 dark:hover:text-primary-300 underline"
 | 
			
		||||
                               href="/admin/user/{{ user.id }}/reset-pw"
 | 
			
		||||
                               onclick="return confirm('Willst du wirklich das Passwort zurücksetzen?');">Passwort zurücksetzen</a>
 | 
			
		||||
                        {% endif %}
 | 
			
		||||
                        <div class="w-full grid gap-3 mt-3">
 | 
			
		||||
                            <input type="hidden" name="id" value="{{ user.id }}" />
 | 
			
		||||
                            <div class="grid sm:grid-cols-2 lg:grid-cols-4 gap-3">
 | 
			
		||||
                                {% for cluster, cluster_roles in roles | group_by(attribute="cluster") %}
 | 
			
		||||
                                    <label for="cluster_{{ loop.index }}">{{ cluster }}</label>
 | 
			
		||||
                                    {# Determine the initially selected role within the cluster #}
 | 
			
		||||
                                    {% set_global selected_role_id = "none" %}
 | 
			
		||||
                                    {% for role in cluster_roles %}
 | 
			
		||||
                                        {% if selected_role_id == "none" and role.name in user.roles %}
 | 
			
		||||
                                            {% set_global selected_role_id = role.id %}
 | 
			
		||||
                                        {% endif %}
 | 
			
		||||
                                    {% endfor %}
 | 
			
		||||
                                    {# Set default name to the selected role ID or first role if none selected #}
 | 
			
		||||
                                    <select id="cluster_{{ loop.index }}"
 | 
			
		||||
                                            {% if selected_role_id == 'none' %} {% else %} name="roles[{{ selected_role_id }}]" {% endif %}
 | 
			
		||||
                                            {% if allowed_to_edit == false %}disabled{% endif %}
 | 
			
		||||
                                            onchange=" if (this.value === '') { this.removeAttribute('name'); } else { this.name = 'roles[' + this.options[this.selectedIndex].getAttribute('data-role-id') + ']'; }">
 | 
			
		||||
                                        <option value=""
 | 
			
		||||
                                                data-role-id="none"
 | 
			
		||||
                                                {% if selected_role_id == 'none' %}selected="selected"{% endif %}>
 | 
			
		||||
                                            None
 | 
			
		||||
                                        </option>
 | 
			
		||||
                                        {% for role in cluster_roles %}
 | 
			
		||||
                                            <option value="on"
 | 
			
		||||
                                                    data-role-id="{{ role.id }}"
 | 
			
		||||
                                                    {% if role.id == selected_role_id %}selected="selected"{% endif %}>
 | 
			
		||||
                                                {{ role.name }}
 | 
			
		||||
                                            </option>
 | 
			
		||||
                                        {% endfor %}
 | 
			
		||||
                                    </select>
 | 
			
		||||
                                {% endfor %}
 | 
			
		||||
                            </div>
 | 
			
		||||
                        </div>
 | 
			
		||||
                    </form>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
 | 
			
		||||
                 role="alert">
 | 
			
		||||
                <h2 class="h2">Ergo-Challenge</h2>
 | 
			
		||||
                <div class="mx-2 divide-y divide-gray-200 dark:divide-primary-600">
 | 
			
		||||
                    <div class="py-3">
 | 
			
		||||
                        {{ macros::input(label='DOB', name='dob', type="text", value=user.dob, readonly=allowed_to_edit == false) }}
 | 
			
		||||
                        {{ macros::input(label='Weight (kg)', name='weight', type="text", value=user.weight, readonly=allowed_to_edit == false) }}
 | 
			
		||||
                        {{ macros::input(label='Sex', name='sex', type="text", value=user.sex, readonly=allowed_to_edit == false) }}
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 
 | 
			
		||||
@@ -174,10 +174,9 @@ function setChoiceByLabel(choicesInstance, label) {
 | 
			
		||||
               {% if autofocus %}autofocus{% endif %}
 | 
			
		||||
               {% if accept %}accept="{{ accept }}"{% endif %}
 | 
			
		||||
               {% if pattern %}pattern="{{ pattern }}"{% endif %}
 | 
			
		||||
               {% if readonly %}readonly{% endif %}/>
 | 
			
		||||
               {% if readonly %}readonly{% endif %} />
 | 
			
		||||
    </div>
 | 
			
		||||
{% endmacro input %}
 | 
			
		||||
 | 
			
		||||
{% macro inputgroup(label, name, type, required=false, class='', value='', min='', hide_label=false, id='', autofocus=false, wrapper_class='', pattern='', readonly=false, accept='') %}
 | 
			
		||||
    <div class="{{ wrapper_class }}">
 | 
			
		||||
        <label for="{{ name }}"
 | 
			
		||||
@@ -185,29 +184,77 @@ function setChoiceByLabel(choicesInstance, label) {
 | 
			
		||||
            {{ label }}
 | 
			
		||||
        </label>
 | 
			
		||||
        <div class="input-group">
 | 
			
		||||
          <input {% if type=='datetime-local' %}onclick='if (!this.value) setCurrentdate(this)'{% endif %}
 | 
			
		||||
                {% if id %} id="{{ id }}" {% else %} id="{{ name }}" {% endif %}
 | 
			
		||||
                name="{{ name }}"
 | 
			
		||||
                type="{{ type }}"
 | 
			
		||||
                {% if required %}required{% endif %}
 | 
			
		||||
                value="{{ value }}"
 | 
			
		||||
                class="input {% if readonly %}rounded-md{% else %}rounded-l-md{% endif %} {{ class }}"
 | 
			
		||||
                placeholder="{% if hide_label %}{{ label }}{% endif %}"
 | 
			
		||||
                {% if min is defined %}min="{{ min }}"{% endif %}
 | 
			
		||||
                {% if autofocus %}autofocus{% endif %}
 | 
			
		||||
                {% if accept %}accept="{{ accept }}"{% endif %}
 | 
			
		||||
                {% if pattern %}pattern="{{ pattern }}"{% endif %}
 | 
			
		||||
                readonly/>
 | 
			
		||||
          {% if allowed_to_edit %}
 | 
			
		||||
              <button type="button" class="btn btn-primary rounded-l-none-important edit-js">Ändern</button>
 | 
			
		||||
              <input value="x" type="reset" class="edit-js btn btn-alert btn-hidden rounded-none-important"/>
 | 
			
		||||
              <input value="💾" type="submit" class="btn btn-primary btn-hidden rounded-l-none-important" />
 | 
			
		||||
          {% endif %}
 | 
			
		||||
            <input {% if type=='datetime-local' %}onclick='if (!this.value) setCurrentdate(this)'{% endif %}
 | 
			
		||||
                   {% if id %} id="{{ id }}" {% else %} id="{{ name }}" {% endif %}
 | 
			
		||||
                   name="{{ name }}"
 | 
			
		||||
                   type="{{ type }}"
 | 
			
		||||
                   {% if required %}required{% endif %}
 | 
			
		||||
                   value="{{ value }}"
 | 
			
		||||
                   class="input {% if readonly %}rounded-md{% else %}rounded-l-md{% endif %} {{ class }}"
 | 
			
		||||
                   placeholder="{% if hide_label %}{{ label }}{% endif %}"
 | 
			
		||||
                   {% if min is defined %}min="{{ min }}"{% endif %}
 | 
			
		||||
                   {% if autofocus %}autofocus{% endif %}
 | 
			
		||||
                   {% if accept %}accept="{{ accept }}"{% endif %}
 | 
			
		||||
                   {% if pattern %}pattern="{{ pattern }}"{% endif %}
 | 
			
		||||
                   readonly />
 | 
			
		||||
            {% if allowed_to_edit %}
 | 
			
		||||
                <button type="button" class="btn btn-dark rounded-l-none-important edit-js">{% include "includes/pencil" %}</button>
 | 
			
		||||
                <input value="x"
 | 
			
		||||
                       type="reset"
 | 
			
		||||
                       class="edit-js btn btn-alert btn-hidden rounded-none-important" />
 | 
			
		||||
                <input value="💾"
 | 
			
		||||
                       type="submit"
 | 
			
		||||
                       class="btn btn-primary btn-hidden rounded-l-none-important" />
 | 
			
		||||
            {% endif %}
 | 
			
		||||
        </div>
 | 
			
		||||
    </div>
 | 
			
		||||
{% endmacro inputgroup %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% macro selectgroup(label, data, name='trip_type', default='', id='', selected_id='', display='', extras='', class='', wrapper_class='', required=false, show_seats=false, new_last_entry='', nonSelectableDefault=false, only_ergo=false, readonly=false) %}
 | 
			
		||||
    <div class="{{ wrapper_class }}">
 | 
			
		||||
        <label for="{{ name }}" class="text-sm text-gray-600 dark:text-gray-100">{{ label }}</label>
 | 
			
		||||
        {% if display == '' %}
 | 
			
		||||
            {% set display = ["name"] %}
 | 
			
		||||
        {% endif %}
 | 
			
		||||
        <div class="input-group">
 | 
			
		||||
            <select name="{{ name }}"
 | 
			
		||||
                    {% if id %} id="{{ id }}" {% else %} id="{{ name }}" {% endif %}
 | 
			
		||||
                    class="input {% if readonly %}rounded-md{% else %}rounded-l-md{% endif %} {{ class }}"
 | 
			
		||||
                    {% if required %}required="required"{% endif %}
 | 
			
		||||
                    disabled>
 | 
			
		||||
                {% if default %}<option selected value>{{ default }}</option>{% endif %}
 | 
			
		||||
                {% if nonSelectableDefault %}<option disabled selected value>{{ nonSelectableDefault }}</option>{% endif %}
 | 
			
		||||
                {% for d in data %}
 | 
			
		||||
                    <option value="{{ d.id }}"
 | 
			
		||||
                            {% if only_ergo and d.id!=4 %}disabled{% endif %}
 | 
			
		||||
                            {% if d.id == selected_id %}selected{% endif %}
 | 
			
		||||
                            {% if extras != '' %} {% for extra in extras %} {% if extra != 'on_water' and d[extra] %} data- {{ extra }}={{ d[extra] }} {% else %} {% if d[extra] %}disabled{% endif %}
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                            {% endfor %}
 | 
			
		||||
                            {% endif %}
 | 
			
		||||
                            {% if show_seats %} data-custom-properties='{"amount_seats": {{ d["amount_seats"] }}, "owner": "{{ d["owner"] }}", "default_destination": "{{ d["default_destination"] }}", "boat_in_ottensheim": {{ d["location_id"] == 2 }}, "boat_reserved_today": {{ d["reserved_today"] }}, "convert_handoperated_possible": {{ d["convert_handoperated_possible"] }}, "default_handoperated": {{ d["default_shipmaster_only_steering"] }}}' {% endif %}>
 | 
			
		||||
                        {% for displa in display -%}
 | 
			
		||||
                            {%- if d[displa] -%}
 | 
			
		||||
                                {{- d[displa] -}}
 | 
			
		||||
                            {%- else -%}
 | 
			
		||||
                                {{- displa -}}
 | 
			
		||||
                            {%- endif -%}
 | 
			
		||||
                        {%- endfor %}
 | 
			
		||||
                    </option>
 | 
			
		||||
                {% endfor %}
 | 
			
		||||
                {% if new_last_entry %}<option value="-1">{{ new_last_entry }}</option>{% endif %}
 | 
			
		||||
            </select>
 | 
			
		||||
            {% if allowed_to_edit %}
 | 
			
		||||
                <button type="button" class="btn btn-dark rounded-l-none-important edit-js">{% include "includes/pencil" %}</button>
 | 
			
		||||
                <input value="x"
 | 
			
		||||
                       type="reset"
 | 
			
		||||
                       class="edit-js btn btn-alert btn-hidden rounded-none-important" />
 | 
			
		||||
                <input value="💾"
 | 
			
		||||
                       type="submit"
 | 
			
		||||
                       class="btn btn-primary btn-hidden rounded-l-none-important" />
 | 
			
		||||
            {% endif %}
 | 
			
		||||
        </div>
 | 
			
		||||
    </div>
 | 
			
		||||
{% endmacro selectgroup %}
 | 
			
		||||
{% macro checkbox(label, name, id='', checked=false, class='', disabled=false, readonly=false) %}
 | 
			
		||||
    <label for="{{ name }}{{ id }}"
 | 
			
		||||
           class="flex items-center cursor-pointer text-black dark:text-white hover:text-gray-900 dark:hover:text-gray-100 {{ class }}">
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user