Compare commits
1 Commits
ce28f93d65
...
improve-lo
Author | SHA1 | Date | |
---|---|---|---|
![]() |
8500ba826f |
@@ -115,7 +115,7 @@ test("Cox can start and finish trip", async ({ page }, testInfo) => {
|
|||||||
await page.getByPlaceholder("Passwort").press("Enter");
|
await page.getByPlaceholder("Passwort").press("Enter");
|
||||||
|
|
||||||
await page.goto("/log/show");
|
await page.goto("/log/show");
|
||||||
await page.getByRole('link', { name: 'Joe' }).nth(1).click();
|
await page.getByText('(cox2)').click();
|
||||||
page.once("dialog", (dialog) => {
|
page.once("dialog", (dialog) => {
|
||||||
dialog.accept().catch(() => {});
|
dialog.accept().catch(() => {});
|
||||||
});
|
});
|
||||||
@@ -208,6 +208,7 @@ test("Kiosk can start and finish trip", async ({ page }, testInfo) => {
|
|||||||
|
|
||||||
await page.getByRole('link', { name: 'Logbuch' }).click();
|
await page.getByRole('link', { name: 'Logbuch' }).click();
|
||||||
await expect(page.locator('body')).toContainText('Joe');
|
await expect(page.locator('body')).toContainText('Joe');
|
||||||
|
await expect(page.locator('body')).toContainText('(cox2)');
|
||||||
await expect(page.locator('body')).toContainText('Ottensheim (25 km)');
|
await expect(page.locator('body')).toContainText('Ottensheim (25 km)');
|
||||||
await expect(page.locator('body')).toContainText('Ruderer: cox2, rower2');
|
await expect(page.locator('body')).toContainText('Ruderer: cox2, rower2');
|
||||||
|
|
||||||
@@ -224,7 +225,7 @@ test("Kiosk can start and finish trip", async ({ page }, testInfo) => {
|
|||||||
await page.getByPlaceholder("Passwort").press("Enter");
|
await page.getByPlaceholder("Passwort").press("Enter");
|
||||||
|
|
||||||
await page.goto("/log/show");
|
await page.goto("/log/show");
|
||||||
await page.getByRole('link', { name: 'Joe' }).nth(1).click();
|
await page.getByText('(cox2)').click();
|
||||||
page.once("dialog", (dialog) => {
|
page.once("dialog", (dialog) => {
|
||||||
dialog.accept().catch(() => {});
|
dialog.accept().catch(() => {});
|
||||||
});
|
});
|
||||||
@@ -285,6 +286,7 @@ test("Cox can start and finish trip with cox steering only", async ({ page }, te
|
|||||||
|
|
||||||
await page.goto('/log/show');
|
await page.goto('/log/show');
|
||||||
await expect(page.locator('body')).toContainText('cox_only_steering_boat');
|
await expect(page.locator('body')).toContainText('cox_only_steering_boat');
|
||||||
|
await expect(page.locator('body')).toContainText('(cox2 - handgesteuert)');
|
||||||
await expect(page.locator('body')).toContainText('Ottensheim (25 km)');
|
await expect(page.locator('body')).toContainText('Ottensheim (25 km)');
|
||||||
|
|
||||||
|
|
||||||
@@ -300,7 +302,7 @@ test("Cox can start and finish trip with cox steering only", async ({ page }, te
|
|||||||
await page.getByPlaceholder("Passwort").press("Enter");
|
await page.getByPlaceholder("Passwort").press("Enter");
|
||||||
|
|
||||||
await page.goto("/log/show");
|
await page.goto("/log/show");
|
||||||
await page.getByRole("link", { name: "cox_only_steering_boat" }).click();
|
await page.getByText('(cox2 - handgesteuert)').click();
|
||||||
page.once("dialog", (dialog) => {
|
page.once("dialog", (dialog) => {
|
||||||
dialog.accept().catch(() => {});
|
dialog.accept().catch(() => {});
|
||||||
});
|
});
|
||||||
@@ -369,7 +371,7 @@ test("Kiosk can start and finish trip in one stop", async ({ page }, testInfo) =
|
|||||||
await page.getByPlaceholder("Passwort").press("Enter");
|
await page.getByPlaceholder("Passwort").press("Enter");
|
||||||
|
|
||||||
await page.goto("/log/show");
|
await page.goto("/log/show");
|
||||||
await page.getByRole('link', { name: 'Joe' }).nth(1).click();
|
await page.getByText('(cox2)').click();
|
||||||
page.once("dialog", (dialog) => {
|
page.once("dialog", (dialog) => {
|
||||||
dialog.accept().catch(() => {});
|
dialog.accept().catch(() => {});
|
||||||
});
|
});
|
||||||
|
@@ -1,21 +1,20 @@
|
|||||||
use std::{fmt::Display, ops::DerefMut};
|
use std::{fmt::Display, ops::DerefMut};
|
||||||
|
|
||||||
use argon2::{Argon2, PasswordHasher, password_hash::SaltString};
|
use argon2::{password_hash::SaltString, Argon2, PasswordHasher};
|
||||||
use chrono::{Datelike, Local, NaiveDate};
|
use chrono::{Datelike, Local, NaiveDate};
|
||||||
use log::info;
|
use log::info;
|
||||||
use rocket::async_trait;
|
use rocket::async_trait;
|
||||||
use rocket::{
|
use rocket::{
|
||||||
Request,
|
|
||||||
http::{Cookie, Status},
|
http::{Cookie, Status},
|
||||||
request::{FromRequest, Outcome},
|
request::{FromRequest, Outcome},
|
||||||
time::{Duration, OffsetDateTime},
|
time::{Duration, OffsetDateTime},
|
||||||
|
Request,
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use sqlx::{FromRow, Sqlite, SqlitePool, Transaction};
|
use sqlx::{FromRow, Sqlite, SqlitePool, Transaction};
|
||||||
|
|
||||||
use super::activity::ActivityBuilder;
|
use super::activity::ActivityBuilder;
|
||||||
use super::{
|
use super::{
|
||||||
Day,
|
|
||||||
log::Log,
|
log::Log,
|
||||||
logbook::Logbook,
|
logbook::Logbook,
|
||||||
mail::Mail,
|
mail::Mail,
|
||||||
@@ -24,6 +23,7 @@ use super::{
|
|||||||
role::Role,
|
role::Role,
|
||||||
stat::Stat,
|
stat::Stat,
|
||||||
tripdetails::TripDetails,
|
tripdetails::TripDetails,
|
||||||
|
Day,
|
||||||
};
|
};
|
||||||
use crate::AMOUNT_DAYS_TO_SHOW_TRIPS_AHEAD;
|
use crate::AMOUNT_DAYS_TO_SHOW_TRIPS_AHEAD;
|
||||||
use scheckbuch::ScheckbuchUser;
|
use scheckbuch::ScheckbuchUser;
|
||||||
@@ -577,6 +577,10 @@ ASKÖ Ruderverein Donau Linz", self.name),
|
|||||||
.execute(db)
|
.execute(db)
|
||||||
.await
|
.await
|
||||||
.unwrap(); //Okay, because we can only create a User of a valid id
|
.unwrap(); //Okay, because we can only create a User of a valid id
|
||||||
|
ActivityBuilder::new(&format!("User {self} hat sich eingeloggt."))
|
||||||
|
.relevant_for_user(self)
|
||||||
|
.save(db)
|
||||||
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn delete(&self, db: &SqlitePool, deleted_by: &ManageUserUser) {
|
pub async fn delete(&self, db: &SqlitePool, deleted_by: &ManageUserUser) {
|
||||||
@@ -863,8 +867,8 @@ special_user!(SteeringUser, +"cox", +"Bootsführer");
|
|||||||
special_user!(AdminUser, +"admin");
|
special_user!(AdminUser, +"admin");
|
||||||
special_user!(AllowedForPlannedTripsUser, +"Donau Linz", +"scheckbuch", +"Förderndes Mitglied");
|
special_user!(AllowedForPlannedTripsUser, +"Donau Linz", +"scheckbuch", +"Förderndes Mitglied");
|
||||||
special_user!(DonauLinzUser, +"Donau Linz", -"Unterstützend", -"Förderndes Mitglied"); // TODO:
|
special_user!(DonauLinzUser, +"Donau Linz", -"Unterstützend", -"Förderndes Mitglied"); // TODO:
|
||||||
// remove ->
|
// remove ->
|
||||||
// RegularUser
|
// RegularUser
|
||||||
special_user!(SchnupperBetreuerUser, +"schnupper-betreuer");
|
special_user!(SchnupperBetreuerUser, +"schnupper-betreuer");
|
||||||
special_user!(VorstandUser, +"admin", +"Vorstand");
|
special_user!(VorstandUser, +"admin", +"Vorstand");
|
||||||
special_user!(EventUser, +"manage_events");
|
special_user!(EventUser, +"manage_events");
|
||||||
@@ -978,21 +982,17 @@ mod test {
|
|||||||
#[sqlx::test]
|
#[sqlx::test]
|
||||||
fn wrong_pw() {
|
fn wrong_pw() {
|
||||||
let pool = testdb!();
|
let pool = testdb!();
|
||||||
assert!(
|
assert!(User::login(&pool, "admin".into(), "admi".into())
|
||||||
User::login(&pool, "admin".into(), "admi".into())
|
|
||||||
.await
|
.await
|
||||||
.is_err()
|
.is_err());
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[sqlx::test]
|
#[sqlx::test]
|
||||||
fn wrong_username() {
|
fn wrong_username() {
|
||||||
let pool = testdb!();
|
let pool = testdb!();
|
||||||
assert!(
|
assert!(User::login(&pool, "admi".into(), "admin".into())
|
||||||
User::login(&pool, "admi".into(), "admin".into())
|
|
||||||
.await
|
.await
|
||||||
.is_err()
|
.is_err());
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[sqlx::test]
|
#[sqlx::test]
|
||||||
@@ -1011,11 +1011,9 @@ mod test {
|
|||||||
let pool = testdb!();
|
let pool = testdb!();
|
||||||
let user = User::find_by_id(&pool, 1).await.unwrap();
|
let user = User::find_by_id(&pool, 1).await.unwrap();
|
||||||
|
|
||||||
assert!(
|
assert!(User::login(&pool, "admin".into(), "abc".into())
|
||||||
User::login(&pool, "admin".into(), "abc".into())
|
|
||||||
.await
|
.await
|
||||||
.is_err()
|
.is_err());
|
||||||
);
|
|
||||||
|
|
||||||
user.update_pw(&pool, "abc".into()).await;
|
user.update_pw(&pool, "abc".into()).await;
|
||||||
|
|
||||||
|
@@ -3,14 +3,13 @@ use crate::model::{
|
|||||||
user::{AdminUser, UserWithDetails, VorstandUser},
|
user::{AdminUser, UserWithDetails, VorstandUser},
|
||||||
};
|
};
|
||||||
use rocket::{
|
use rocket::{
|
||||||
FromForm, Route, State,
|
|
||||||
form::Form,
|
form::Form,
|
||||||
get, post,
|
get, post,
|
||||||
request::FlashMessage,
|
request::FlashMessage,
|
||||||
response::{Flash, Redirect},
|
response::{Flash, Redirect},
|
||||||
routes,
|
routes, FromForm, Route, State,
|
||||||
};
|
};
|
||||||
use rocket_dyn_templates::{Template, tera::Context};
|
use rocket_dyn_templates::{tera::Context, Template};
|
||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
|
|
||||||
#[get("/role")]
|
#[get("/role")]
|
||||||
|
@@ -7,11 +7,11 @@ use crate::{
|
|||||||
mail::valid_mails,
|
mail::valid_mails,
|
||||||
role::Role,
|
role::Role,
|
||||||
user::{
|
user::{
|
||||||
AdminUser, AllowedToEditPaymentStatusUser, ManageUserUser, User, UserWithDetails,
|
|
||||||
UserWithMembershipPdf, UserWithRolesAndMembershipPdf, VorstandUser,
|
|
||||||
clubmember::ClubMemberUser, foerdernd::FoerderndUser, member::Member,
|
clubmember::ClubMemberUser, foerdernd::FoerderndUser, member::Member,
|
||||||
regular::RegularUser, scheckbuch::ScheckbuchUser, schnupperant::SchnupperantUser,
|
regular::RegularUser, scheckbuch::ScheckbuchUser, schnupperant::SchnupperantUser,
|
||||||
schnupperinterest::SchnupperInterestUser, unterstuetzend::UnterstuetzendUser,
|
schnupperinterest::SchnupperInterestUser, unterstuetzend::UnterstuetzendUser,
|
||||||
|
AdminUser, AllowedToEditPaymentStatusUser, ManageUserUser, User, UserWithDetails,
|
||||||
|
UserWithMembershipPdf, UserWithRolesAndMembershipPdf, VorstandUser,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
tera::Config,
|
tera::Config,
|
||||||
@@ -19,7 +19,6 @@ use crate::{
|
|||||||
use chrono::NaiveDate;
|
use chrono::NaiveDate;
|
||||||
use futures::future::join_all;
|
use futures::future::join_all;
|
||||||
use rocket::{
|
use rocket::{
|
||||||
FromForm, Request, Route, State,
|
|
||||||
form::Form,
|
form::Form,
|
||||||
fs::TempFile,
|
fs::TempFile,
|
||||||
get,
|
get,
|
||||||
@@ -27,9 +26,9 @@ use rocket::{
|
|||||||
post,
|
post,
|
||||||
request::{FlashMessage, FromRequest, Outcome},
|
request::{FlashMessage, FromRequest, Outcome},
|
||||||
response::{Flash, Redirect},
|
response::{Flash, Redirect},
|
||||||
routes,
|
routes, FromForm, Request, Route, State,
|
||||||
};
|
};
|
||||||
use rocket_dyn_templates::{Template, tera::Context};
|
use rocket_dyn_templates::{tera::Context, Template};
|
||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
|
|
||||||
// Custom request guard to extract the Referer header
|
// Custom request guard to extract the Referer header
|
||||||
@@ -136,7 +135,7 @@ async fn view(
|
|||||||
if user.name == "Externe Steuerperson" {
|
if user.name == "Externe Steuerperson" {
|
||||||
return Err(Flash::error(
|
return Err(Flash::error(
|
||||||
Redirect::to("/admin/user"),
|
Redirect::to("/admin/user"),
|
||||||
"Diese besondere Person kannst du dir leider nicht anschauen, mein lieber neugieriger Ruderant!",
|
"Diese besondere Person kannst du dir leider nicht anschauen, mein lieber neugieriger Ruderant!"
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -14,7 +14,6 @@ use rocket_dyn_templates::{Template, context, tera};
|
|||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
|
|
||||||
use crate::model::{
|
use crate::model::{
|
||||||
activity::ActivityBuilder,
|
|
||||||
log::Log,
|
log::Log,
|
||||||
user::{LoginError, User},
|
user::{LoginError, User},
|
||||||
};
|
};
|
||||||
@@ -83,12 +82,13 @@ async fn login(
|
|||||||
|
|
||||||
cookies.add_private(Cookie::new("loggedin_user", format!("{}", user.id)));
|
cookies.add_private(Cookie::new("loggedin_user", format!("{}", user.id)));
|
||||||
|
|
||||||
ActivityBuilder::new(&format!(
|
Log::create(
|
||||||
"{user} hat sich eingeloggt (User-Agent: {})",
|
db,
|
||||||
agent.0
|
format!(
|
||||||
))
|
"Succ login of {} with this useragent: {}",
|
||||||
.relevant_for_user(&user)
|
login.name, agent.0
|
||||||
.save(db)
|
),
|
||||||
|
)
|
||||||
.await;
|
.await;
|
||||||
|
|
||||||
// Check for redirect_url cookie and redirect accordingly
|
// Check for redirect_url cookie and redirect accordingly
|
||||||
|
@@ -1,7 +1,6 @@
|
|||||||
use std::net::IpAddr;
|
use std::net::IpAddr;
|
||||||
|
|
||||||
use rocket::{
|
use rocket::{
|
||||||
Request, Route, State,
|
|
||||||
form::Form,
|
form::Form,
|
||||||
get,
|
get,
|
||||||
http::{Cookie, CookieJar},
|
http::{Cookie, CookieJar},
|
||||||
@@ -10,8 +9,9 @@ use rocket::{
|
|||||||
response::{Flash, Redirect},
|
response::{Flash, Redirect},
|
||||||
routes,
|
routes,
|
||||||
time::{Duration, OffsetDateTime},
|
time::{Duration, OffsetDateTime},
|
||||||
|
Request, Route, State,
|
||||||
};
|
};
|
||||||
use rocket_dyn_templates::{Template, context};
|
use rocket_dyn_templates::{context, Template};
|
||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
use tera::Context;
|
use tera::Context;
|
||||||
|
|
||||||
@@ -585,7 +585,7 @@ mod test {
|
|||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
|
|
||||||
use crate::model::logbook::Logbook;
|
use crate::model::logbook::Logbook;
|
||||||
use crate::tera::{User, log::Boat};
|
use crate::tera::{log::Boat, User};
|
||||||
use crate::testdb;
|
use crate::testdb;
|
||||||
|
|
||||||
#[sqlx::test]
|
#[sqlx::test]
|
||||||
|
@@ -8,16 +8,19 @@
|
|||||||
<summary class="px-3 cursor-pointer text-md font-bold text-primary-950 dark:text-white">
|
<summary class="px-3 cursor-pointer text-md font-bold text-primary-950 dark:text-white">
|
||||||
Neue Person hinzufügen
|
Neue Person hinzufügen
|
||||||
</summary>
|
</summary>
|
||||||
|
|
||||||
<div class="grid sm:grid-cols-3 gap-3 mt-3">
|
<div class="grid sm:grid-cols-3 gap-3 mt-3">
|
||||||
<button type="button"
|
<button type="button"
|
||||||
onclick="document.getElementById('add-clubuser').showModal()"
|
onclick="document.getElementById('add-clubuser').showModal()"
|
||||||
class="btn btn-primary">🥳 Vereinsmitglied</button>
|
class="btn btn-primary">Vereinsmitglied</button>
|
||||||
<button type="button"
|
<button type="button"
|
||||||
onclick="document.getElementById('add-scheckbuch').showModal()"
|
onclick="document.getElementById('add-scheckbuch').showModal()"
|
||||||
class="btn btn-dark">🧑🏫 Scheckbuch</button>
|
class="btn btn-dark">Scheckbuch</button>
|
||||||
<button type="button"
|
<button type="button"
|
||||||
onclick="document.getElementById('add-schnupperkurs').showModal()"
|
onclick="document.getElementById('add-schnupperkurs').showModal()"
|
||||||
class="btn btn-dark">👨🎓 Schnupperkurs</button>
|
class="btn btn-dark">Schnupperkurs</button>
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<dialog id="add-clubuser"
|
<dialog id="add-clubuser"
|
||||||
class="max-w-screen-sm w-full dark:bg-primary-900 dark:text-white rounded-md"
|
class="max-w-screen-sm w-full dark:bg-primary-900 dark:text-white rounded-md"
|
||||||
@@ -64,6 +67,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</dialog>
|
</dialog>
|
||||||
|
|
||||||
<dialog id="add-scheckbuch"
|
<dialog id="add-scheckbuch"
|
||||||
class="max-w-screen-sm w-full dark:bg-primary-900 dark:text-white rounded-md"
|
class="max-w-screen-sm w-full dark:bg-primary-900 dark:text-white rounded-md"
|
||||||
onclick="document.getElementById('add-scheckbuch').close()">
|
onclick="document.getElementById('add-scheckbuch').close()">
|
||||||
@@ -95,6 +99,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</dialog>
|
</dialog>
|
||||||
|
|
||||||
<dialog id="add-schnupperkurs"
|
<dialog id="add-schnupperkurs"
|
||||||
class="max-w-screen-sm w-full dark:bg-primary-900 dark:text-white rounded-md"
|
class="max-w-screen-sm w-full dark:bg-primary-900 dark:text-white rounded-md"
|
||||||
onclick="document.getElementById('add-schnupperkurs').close()">
|
onclick="document.getElementById('add-schnupperkurs').close()">
|
||||||
@@ -117,6 +122,7 @@
|
|||||||
enctype="multipart/form-data"
|
enctype="multipart/form-data"
|
||||||
class="grid gap-3">
|
class="grid gap-3">
|
||||||
<h2 class="h3 mb-3">Neuer Schnupperant</h2>
|
<h2 class="h3 mb-3">Neuer Schnupperant</h2>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label for="schnupper_type" class="text-sm text-gray-600 dark:text-gray-100">Typ</label>
|
<label for="schnupper_type" class="text-sm text-gray-600 dark:text-gray-100">Typ</label>
|
||||||
<select name="schnupper_type" id="schnupper_type" class="input rounded-md ">
|
<select name="schnupper_type" id="schnupper_type" class="input rounded-md ">
|
||||||
|
@@ -4,9 +4,7 @@
|
|||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="max-w-screen-lg w-full">
|
<div class="max-w-screen-lg w-full">
|
||||||
{% if "admin" in loggedin_user.roles or "Vorstand" in loggedin_user.roles %}
|
{% if "admin" in loggedin_user.roles or "Vorstand" in loggedin_user.roles %}
|
||||||
<div class="mb-5 lg:mb-0">
|
|
||||||
<a href="/admin/user" class="link link-primary link-no-underline">← Userverwaltung</a>
|
<a href="/admin/user" class="link link-primary link-no-underline">← Userverwaltung</a>
|
||||||
</div>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<h1 class="h1">{{ user.name }}</h1>
|
<h1 class="h1">{{ user.name }}</h1>
|
||||||
<div class="grid sm:grid-cols-2 gap-8 my-8">
|
<div class="grid sm:grid-cols-2 gap-8 my-8">
|
||||||
@@ -121,12 +119,12 @@
|
|||||||
</div>
|
</div>
|
||||||
{% if allowed_to_edit %}
|
{% if allowed_to_edit %}
|
||||||
<div class="py-3">
|
<div class="py-3">
|
||||||
<div class="text-right">
|
<div class="mt-3 text-right">
|
||||||
<button type="button"
|
<button type="button"
|
||||||
onclick="document.getElementById('change-member-type').showModal()"
|
onclick="document.getElementById('change-member-type').showModal()"
|
||||||
class="btn btn-dark">Mitgliedsstatus ändern</button>
|
class="btn btn-dark">Mitgliedsstatus ändern</button>
|
||||||
<a href="/admin/user/{{ user.id }}/delete"
|
<a href="/admin/user/{{ user.id }}/delete"
|
||||||
class="btn btn-alert mt-3"
|
class="btn btn-alert"
|
||||||
onclick="return confirm('Ist {{ user.name }} wirklich aus dem Verein ausgetreten?');">
|
onclick="return confirm('Ist {{ user.name }} wirklich aus dem Verein ausgetreten?');">
|
||||||
{% include "includes/delete-icon" %}
|
{% include "includes/delete-icon" %}
|
||||||
Mitglied ist ausgetreten
|
Mitglied ist ausgetreten
|
||||||
@@ -387,11 +385,9 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
{% else %}
|
{% else %}
|
||||||
{% if "paid" in user.roles %}
|
{% if "paid" in user.roles %}
|
||||||
✅
|
✅ {% for key, value in member %}
|
||||||
{% for key, value in member %}
|
|
||||||
{% if loop.first %}{{ key }}{% endif %}
|
{% if loop.first %}{{ key }}{% endif %}
|
||||||
{% endfor %}
|
{% endfor %} hat schon bezahlt
|
||||||
hat schon bezahlt
|
|
||||||
{% else %}
|
{% else %}
|
||||||
❌
|
❌
|
||||||
{% for key, value in member %}
|
{% for key, value in member %}
|
||||||
@@ -406,13 +402,11 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow">
|
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow">
|
||||||
<h2 class="h2">Aktivitäten</h2>
|
<h2 class="h2">Aktivitäten</h2>
|
||||||
<div class="mx-3 max-h-60 overflow-y-scroll">
|
<div class="mx-3 divide-y divide-gray-200 dark:divide-primary-600">
|
||||||
<div class="py-3">
|
<div class="py-3">
|
||||||
<ul class="list-disc ms-4">
|
<ul class="list-disc ms-4">
|
||||||
{% for activity in activities %}
|
{% for activity in activities %}
|
||||||
<li>
|
<li>{{ activity.created_at | date(format="%d. %m. %Y") }}: {{ activity.text }}</li>
|
||||||
<strong>{{ activity.created_at | date(format="%d. %m. %Y") }}:</strong> <small>{{ activity.text }}</small>
|
|
||||||
</li>
|
|
||||||
{% else %}
|
{% else %}
|
||||||
<li>Noch keine Aktivität... Stay tuned 😆</li>
|
<li>Noch keine Aktivität... Stay tuned 😆</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@@ -202,7 +202,9 @@
|
|||||||
onclick="document.getElementById('change-{{ log.id }}').showModal()"
|
onclick="document.getElementById('change-{{ log.id }}').showModal()"
|
||||||
class="link link-black font-bold">{{ log.boat.name }}</a>
|
class="link link-black font-bold">{{ log.boat.name }}</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<strong class="text-black dark:text-white">{{ log.boat.name }}</strong>
|
<strong class="text-black dark:text-white">
|
||||||
|
{{ log.boat.name }}
|
||||||
|
</strong>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<small class="text-gray-600 dark:text-gray-100">({{ log.shipmaster_user.name -}}
|
<small class="text-gray-600 dark:text-gray-100">({{ log.shipmaster_user.name -}}
|
||||||
{% if log.shipmaster_only_steering %}
|
{% if log.shipmaster_only_steering %}
|
||||||
@@ -274,24 +276,31 @@
|
|||||||
</svg>
|
</svg>
|
||||||
</button>
|
</button>
|
||||||
<div class="mt-8">
|
<div class="mt-8">
|
||||||
<h2 class="h3">Eintrag '{{ log.boat.name }}' ändern</h2>
|
<h2 class="h3">Eintrag '{{ log.boat.name }}' ändern </h2>
|
||||||
<p class="text-center mb-3">{{ log.id }}</p>
|
<p class="text-center mb-3">ID: {{ log.id }}</p>
|
||||||
<form action="/log/update" method="post" class="grid gap-3">
|
<form action="/log/update" method="post" class="grid gap-3">
|
||||||
<input type="hidden" name="id" value="{{ log.id }}" />
|
<input type="hidden" name="id" value="{{ log.id }}" />
|
||||||
<input type="hidden" name="boat_id" value="{{ log.boat_id }}" />
|
|
||||||
<input type="hidden" name="shipmaster" value="{{ log.shipmaster }}" />
|
|
||||||
<input type="hidden"
|
<input type="hidden"
|
||||||
name="steering_person"
|
name="steering_person"
|
||||||
value="{{ log.steering_person }}" />
|
value="{{ log.steering_person }}" />
|
||||||
|
{{ macros::select(label="Boot", data=boats, name="boat_id", id="boat_id{{ log.id }}", selected_id=log.boat.id ,display=["name", " (","amount_seats", " x)"]) }}
|
||||||
|
{{ macros::select(label="Schiffsführer", data=log.rowers, name="shipmaster", id="shipmaster{{ log.id }}", selected_id=log.shipmaster_user.id) }}
|
||||||
|
|
||||||
|
|
||||||
{{ macros::checkbox(label='Handgesteuert', name='shipmaster_only_steering', id=log.shipmaster_only_steering,checked=log.shipmaster_only_steering) }}
|
{{ macros::checkbox(label='Handgesteuert', name='shipmaster_only_steering', id=log.shipmaster_only_steering,checked=log.shipmaster_only_steering) }}
|
||||||
<input type="datetime-local"
|
<div>
|
||||||
class="input rounded-md"
|
<label for="departure" class=" text-sm text-gray-600 dark:text-white ">
|
||||||
name="departure"
|
Abfahrt
|
||||||
value="{{ log.departure }}" />
|
</label>
|
||||||
<input type="datetime-local"
|
<input type="datetime-local" class="input rounded-md" name="departure" value="{{ log.departure }}" />
|
||||||
class="input rounded-md"
|
</div>
|
||||||
name="arrival"
|
<div>
|
||||||
value="{{ log.arrival }}" />
|
<label for="arrival" class=" text-sm text-gray-600 dark:text-white ">
|
||||||
|
Ankunft
|
||||||
|
</label>
|
||||||
|
<input type="datetime-local" class="input rounded-md" name="arrival" value="{{ log.arrival }}" />
|
||||||
|
</div>
|
||||||
|
|
||||||
<input type="hidden" name="destination" value="{{ log.destination }}" />
|
<input type="hidden" name="destination" value="{{ log.destination }}" />
|
||||||
<input type="hidden" name="distance_in_km" value="{{ log.distance_in_km }}" />
|
<input type="hidden" name="distance_in_km" value="{{ log.distance_in_km }}" />
|
||||||
<input type="hidden" name="comments" value="{{ log.comments }}" />
|
<input type="hidden" name="comments" value="{{ log.comments }}" />
|
||||||
|
@@ -26,7 +26,7 @@
|
|||||||
{% for log in logs %}
|
{% for log in logs %}
|
||||||
{% set_global allowed_to_edit = false %}
|
{% set_global allowed_to_edit = false %}
|
||||||
{% if loggedin_user %}
|
{% if loggedin_user %}
|
||||||
{% if "Vorstand" in loggedin_user.roles or "admin" in loggedin_user.roles %}
|
{% if "Vorstand" in loggedin_user.roles %}
|
||||||
{% set_global allowed_to_edit = true %}
|
{% set_global allowed_to_edit = true %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
Reference in New Issue
Block a user