Compare commits
15 Commits
e7a679541b
...
b0ea0668c7
Author | SHA1 | Date | |
---|---|---|---|
b0ea0668c7 | |||
244eb1be07 | |||
784deaf9f4 | |||
3b63cafa79 | |||
0fe672c9da | |||
37ab6e9132 | |||
53afb4ee6f | |||
1783527f39 | |||
32b2185e94 | |||
6fb27d52d6 | |||
092dba3f4b | |||
4466c9f018 | |||
0ba2590bfd | |||
cccf62bb53 | |||
649169c192 |
@ -7,6 +7,7 @@ export interface choiceMap {
|
|||||||
|
|
||||||
let choiceObjects: choiceMap = {};
|
let choiceObjects: choiceMap = {};
|
||||||
let boat_in_ottensheim = true;
|
let boat_in_ottensheim = true;
|
||||||
|
let boat_reserved_today= true;
|
||||||
|
|
||||||
document.addEventListener("DOMContentLoaded", function () {
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
changeTheme();
|
changeTheme();
|
||||||
@ -116,6 +117,7 @@ interface ChoiceBoatEvent extends Event {
|
|||||||
owner: number;
|
owner: number;
|
||||||
default_destination: string;
|
default_destination: string;
|
||||||
boat_in_ottensheim: boolean;
|
boat_in_ottensheim: boolean;
|
||||||
|
boat_reserved_today: boolean;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -134,7 +136,12 @@ function selectBoatChange() {
|
|||||||
boatSelect.addEventListener(
|
boatSelect.addEventListener(
|
||||||
"addItem",
|
"addItem",
|
||||||
function (e) {
|
function (e) {
|
||||||
|
|
||||||
const event = e as ChoiceBoatEvent;
|
const event = e as ChoiceBoatEvent;
|
||||||
|
boat_reserved_today = event.detail.customProperties.boat_reserved_today;
|
||||||
|
if (boat_reserved_today){
|
||||||
|
alert(event.detail.label.trim()+' wurde heute reserviert. Bitte kontrolliere, dass du die Reservierung nicht störst.');
|
||||||
|
}
|
||||||
boat_in_ottensheim = event.detail.customProperties.boat_in_ottensheim;
|
boat_in_ottensheim = event.detail.customProperties.boat_in_ottensheim;
|
||||||
|
|
||||||
const amount_seats = event.detail.customProperties.amount_seats;
|
const amount_seats = event.detail.customProperties.amount_seats;
|
||||||
|
@ -160,3 +160,16 @@ CREATE TABLE IF NOT EXISTS "notification" (
|
|||||||
"category" TEXT NOT NULL,
|
"category" TEXT NOT NULL,
|
||||||
"link" TEXT
|
"link" TEXT
|
||||||
);
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS "boat_reservation" (
|
||||||
|
"id" integer NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||||
|
"boat_id" INTEGER NOT NULL REFERENCES boat(id),
|
||||||
|
"start_date" DATE NOT NULL,
|
||||||
|
"end_date" DATE NOT NULL,
|
||||||
|
"time_desc" TEXT NOT NULL,
|
||||||
|
"usage" TEXT NOT NULL,
|
||||||
|
"user_id_applicant" INTEGER NOT NULL REFERENCES user(id),
|
||||||
|
"user_id_confirmation" INTEGER REFERENCES user(id),
|
||||||
|
"created_at" datetime not null default CURRENT_TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
@ -36,9 +36,10 @@ pub enum BoatDamage {
|
|||||||
#[derive(Serialize, Deserialize, Debug)]
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
pub struct BoatWithDetails {
|
pub struct BoatWithDetails {
|
||||||
#[serde(flatten)]
|
#[serde(flatten)]
|
||||||
boat: Boat,
|
pub(crate) boat: Boat,
|
||||||
damage: BoatDamage,
|
damage: BoatDamage,
|
||||||
on_water: bool,
|
on_water: bool,
|
||||||
|
reserved_today: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(FromForm)]
|
#[derive(FromForm)]
|
||||||
@ -135,6 +136,20 @@ impl Boat {
|
|||||||
sqlx::query!("SELECT * FROM boat_damage WHERE boat_id=? AND lock_boat=false AND user_id_verified is null", self.id).fetch_optional(db).await.unwrap().is_some()
|
sqlx::query!("SELECT * FROM boat_damage WHERE boat_id=? AND lock_boat=false AND user_id_verified is null", self.id).fetch_optional(db).await.unwrap().is_some()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn reserved_today(&self, db: &SqlitePool) -> bool {
|
||||||
|
sqlx::query!(
|
||||||
|
"SELECT *
|
||||||
|
FROM boat_reservation
|
||||||
|
WHERE boat_id =?
|
||||||
|
AND date('now') BETWEEN start_date AND end_date;",
|
||||||
|
self.id
|
||||||
|
)
|
||||||
|
.fetch_optional(db)
|
||||||
|
.await
|
||||||
|
.unwrap()
|
||||||
|
.is_some()
|
||||||
|
}
|
||||||
|
|
||||||
pub async fn on_water(&self, db: &SqlitePool) -> bool {
|
pub async fn on_water(&self, db: &SqlitePool) -> bool {
|
||||||
sqlx::query!(
|
sqlx::query!(
|
||||||
"SELECT * FROM logbook WHERE boat_id=? AND arrival is null",
|
"SELECT * FROM logbook WHERE boat_id=? AND arrival is null",
|
||||||
@ -159,6 +174,7 @@ impl Boat {
|
|||||||
res.push(BoatWithDetails {
|
res.push(BoatWithDetails {
|
||||||
damage,
|
damage,
|
||||||
on_water: boat.on_water(db).await,
|
on_water: boat.on_water(db).await,
|
||||||
|
reserved_today: boat.reserved_today(db).await,
|
||||||
boat,
|
boat,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
178
src/model/boatreservation.rs
Normal file
178
src/model/boatreservation.rs
Normal file
@ -0,0 +1,178 @@
|
|||||||
|
use crate::model::{boat::Boat, user::User};
|
||||||
|
use chrono::NaiveDate;
|
||||||
|
use chrono::NaiveDateTime;
|
||||||
|
use rocket::serde::{Deserialize, Serialize};
|
||||||
|
use sqlx::{FromRow, SqlitePool};
|
||||||
|
|
||||||
|
use super::log::Log;
|
||||||
|
use super::notification::Notification;
|
||||||
|
use super::role::Role;
|
||||||
|
|
||||||
|
#[derive(FromRow, Debug, Serialize, Deserialize)]
|
||||||
|
pub struct BoatReservation {
|
||||||
|
pub id: i64,
|
||||||
|
pub boat_id: i64,
|
||||||
|
pub start_date: NaiveDate,
|
||||||
|
pub end_date: NaiveDate,
|
||||||
|
pub time_desc: String,
|
||||||
|
pub usage: String,
|
||||||
|
pub user_id_applicant: i64,
|
||||||
|
pub user_id_confirmation: Option<i64>,
|
||||||
|
pub created_at: NaiveDateTime,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(FromRow, Debug, Serialize, Deserialize)]
|
||||||
|
pub struct BoatReservationWithDetails {
|
||||||
|
#[serde(flatten)]
|
||||||
|
boat_reservation: BoatReservation,
|
||||||
|
boat: Boat,
|
||||||
|
user_applicant: User,
|
||||||
|
user_confirmation: Option<User>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct BoatReservationToAdd<'r> {
|
||||||
|
pub boat: &'r Boat,
|
||||||
|
pub start_date: NaiveDate,
|
||||||
|
pub end_date: NaiveDate,
|
||||||
|
pub time_desc: &'r str,
|
||||||
|
pub usage: &'r str,
|
||||||
|
pub user_applicant: &'r User,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BoatReservation {
|
||||||
|
pub async fn find_by_id(db: &SqlitePool, id: i32) -> Option<Self> {
|
||||||
|
sqlx::query_as!(
|
||||||
|
Self,
|
||||||
|
"SELECT id, boat_id, start_date, end_date, time_desc, usage, user_id_applicant, user_id_confirmation, created_at
|
||||||
|
FROM boat_reservation
|
||||||
|
WHERE id like ?",
|
||||||
|
id
|
||||||
|
)
|
||||||
|
.fetch_one(db)
|
||||||
|
.await
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn all_future(db: &SqlitePool) -> Vec<BoatReservationWithDetails> {
|
||||||
|
let boatreservations = sqlx::query_as!(
|
||||||
|
Self,
|
||||||
|
"
|
||||||
|
SELECT id, boat_id, start_date, end_date, time_desc, usage, user_id_applicant, user_id_confirmation, created_at
|
||||||
|
FROM boat_reservation
|
||||||
|
WHERE end_date >= CURRENT_DATE ORDER BY end_date
|
||||||
|
"
|
||||||
|
)
|
||||||
|
.fetch_all(db)
|
||||||
|
.await
|
||||||
|
.unwrap(); //TODO: fixme
|
||||||
|
|
||||||
|
let mut res = Vec::new();
|
||||||
|
for reservation in boatreservations {
|
||||||
|
let user_confirmation = match reservation.user_id_confirmation {
|
||||||
|
Some(id) => {
|
||||||
|
let user = User::find_by_id(db, id as i32).await;
|
||||||
|
Some(user.unwrap())
|
||||||
|
}
|
||||||
|
None => None,
|
||||||
|
};
|
||||||
|
let user_applicant = User::find_by_id(db, reservation.user_id_applicant as i32)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
let boat = Boat::find_by_id(db, reservation.boat_id as i32)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
res.push(BoatReservationWithDetails {
|
||||||
|
boat_reservation: reservation,
|
||||||
|
boat,
|
||||||
|
user_applicant,
|
||||||
|
user_confirmation,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn create(
|
||||||
|
db: &SqlitePool,
|
||||||
|
boatreservation: BoatReservationToAdd<'_>,
|
||||||
|
) -> Result<(), String> {
|
||||||
|
if Self::boat_reserved_between_dates(
|
||||||
|
db,
|
||||||
|
boatreservation.boat,
|
||||||
|
&boatreservation.start_date,
|
||||||
|
&boatreservation.end_date,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
return Err("Boot in diesem Zeitraum bereits reserviert.".into());
|
||||||
|
}
|
||||||
|
|
||||||
|
Log::create(db, format!("New boat reservation: {boatreservation:?}")).await;
|
||||||
|
|
||||||
|
sqlx::query!(
|
||||||
|
"INSERT INTO boat_reservation(boat_id, start_date, end_date, time_desc, usage, user_id_applicant) VALUES (?,?,?,?,?,?)",
|
||||||
|
boatreservation.boat.id,
|
||||||
|
boatreservation.start_date,
|
||||||
|
boatreservation.end_date,
|
||||||
|
boatreservation.time_desc,
|
||||||
|
boatreservation.usage,
|
||||||
|
boatreservation.user_applicant.id,
|
||||||
|
)
|
||||||
|
.execute(db)
|
||||||
|
.await
|
||||||
|
.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
|
let board =
|
||||||
|
User::all_with_role(db, &Role::find_by_name(db, "Vorstand").await.unwrap()).await;
|
||||||
|
for user in board {
|
||||||
|
Notification::create(
|
||||||
|
db,
|
||||||
|
&user,
|
||||||
|
&format!(
|
||||||
|
"{} hat eine neue Bootsreservierung für Boot '{}' angelegt: Von {} bis {} um {} wegen {}",
|
||||||
|
boatreservation.user_applicant.name,
|
||||||
|
boatreservation.boat.name,
|
||||||
|
boatreservation.start_date,
|
||||||
|
boatreservation.end_date,
|
||||||
|
boatreservation.time_desc,
|
||||||
|
boatreservation.usage
|
||||||
|
),
|
||||||
|
"Neue Bootsreservierung",
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn boat_reserved_between_dates(
|
||||||
|
db: &SqlitePool,
|
||||||
|
boat: &Boat,
|
||||||
|
start_date: &NaiveDate,
|
||||||
|
end_date: &NaiveDate,
|
||||||
|
) -> bool {
|
||||||
|
sqlx::query!(
|
||||||
|
"SELECT COUNT(*) AS reservation_count
|
||||||
|
FROM boat_reservation
|
||||||
|
WHERE boat_id = ?
|
||||||
|
AND start_date <= ? AND end_date >= ?;",
|
||||||
|
boat.id,
|
||||||
|
end_date,
|
||||||
|
start_date
|
||||||
|
)
|
||||||
|
.fetch_one(db)
|
||||||
|
.await
|
||||||
|
.unwrap()
|
||||||
|
.reservation_count
|
||||||
|
> 0
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn delete(&self, db: &SqlitePool) {
|
||||||
|
sqlx::query!("DELETE FROM boat_reservation WHERE id=?", self.id)
|
||||||
|
.execute(db)
|
||||||
|
.await
|
||||||
|
.unwrap(); //Okay, because we can only create a Boat of a valid id
|
||||||
|
}
|
||||||
|
}
|
@ -10,6 +10,7 @@ use self::{
|
|||||||
pub mod boat;
|
pub mod boat;
|
||||||
pub mod boatdamage;
|
pub mod boatdamage;
|
||||||
pub mod boathouse;
|
pub mod boathouse;
|
||||||
|
pub mod boatreservation;
|
||||||
pub mod family;
|
pub mod family;
|
||||||
pub mod location;
|
pub mod location;
|
||||||
pub mod log;
|
pub mod log;
|
||||||
|
@ -6,7 +6,7 @@ use ics::{
|
|||||||
Event, ICalendar,
|
Event, ICalendar,
|
||||||
};
|
};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use sqlx::{FromRow, SqlitePool, Row};
|
use sqlx::{FromRow, Row, SqlitePool};
|
||||||
|
|
||||||
use super::{tripdetails::TripDetails, triptype::TripType, user::User};
|
use super::{tripdetails::TripDetails, triptype::TripType, user::User};
|
||||||
|
|
||||||
|
171
src/tera/boatreservation.rs
Normal file
171
src/tera/boatreservation.rs
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
use chrono::NaiveDate;
|
||||||
|
use rocket::{
|
||||||
|
form::Form,
|
||||||
|
get, post,
|
||||||
|
request::FlashMessage,
|
||||||
|
response::{Flash, Redirect},
|
||||||
|
routes, FromForm, Route, State,
|
||||||
|
};
|
||||||
|
use rocket_dyn_templates::Template;
|
||||||
|
use sqlx::SqlitePool;
|
||||||
|
use tera::Context;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
model::{
|
||||||
|
boat::Boat,
|
||||||
|
boatreservation::{BoatReservation, BoatReservationToAdd},
|
||||||
|
user::{DonauLinzUser, User, UserWithRoles},
|
||||||
|
},
|
||||||
|
tera::log::KioskCookie,
|
||||||
|
};
|
||||||
|
|
||||||
|
#[get("/")]
|
||||||
|
async fn index_kiosk(
|
||||||
|
db: &State<SqlitePool>,
|
||||||
|
flash: Option<FlashMessage<'_>>,
|
||||||
|
_kiosk: KioskCookie,
|
||||||
|
) -> Template {
|
||||||
|
let boatreservations = BoatReservation::all_future(db).await;
|
||||||
|
|
||||||
|
let mut context = Context::new();
|
||||||
|
if let Some(msg) = flash {
|
||||||
|
context.insert("flash", &msg.into_inner());
|
||||||
|
}
|
||||||
|
|
||||||
|
let linz_boats = Boat::all_for_boatshouse(db).await;
|
||||||
|
let mut boats = Vec::new();
|
||||||
|
for boat in linz_boats {
|
||||||
|
if boat.boat.owner.is_none() {
|
||||||
|
boats.push(boat);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
context.insert("boatreservations", &boatreservations);
|
||||||
|
context.insert("boats", &boats);
|
||||||
|
context.insert("user", &User::all(db).await);
|
||||||
|
context.insert("show_kiosk_header", &true);
|
||||||
|
|
||||||
|
Template::render("boatreservations", context.into_json())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[get("/", rank = 2)]
|
||||||
|
async fn index(
|
||||||
|
db: &State<SqlitePool>,
|
||||||
|
flash: Option<FlashMessage<'_>>,
|
||||||
|
user: DonauLinzUser,
|
||||||
|
) -> Template {
|
||||||
|
let boatreservations = BoatReservation::all_future(db).await;
|
||||||
|
|
||||||
|
let mut context = Context::new();
|
||||||
|
if let Some(msg) = flash {
|
||||||
|
context.insert("flash", &msg.into_inner());
|
||||||
|
}
|
||||||
|
|
||||||
|
let linz_boats = Boat::all_for_boatshouse(db).await;
|
||||||
|
let mut boats = Vec::new();
|
||||||
|
for boat in linz_boats {
|
||||||
|
if boat.boat.owner.is_none() {
|
||||||
|
boats.push(boat);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
context.insert("boatreservations", &boatreservations);
|
||||||
|
context.insert("boats", &boats);
|
||||||
|
context.insert("user", &User::all(db).await);
|
||||||
|
context.insert(
|
||||||
|
"loggedin_user",
|
||||||
|
&UserWithRoles::from_user(user.into(), db).await,
|
||||||
|
);
|
||||||
|
|
||||||
|
Template::render("boatreservations", context.into_json())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, FromForm)]
|
||||||
|
pub struct FormBoatReservationToAdd<'r> {
|
||||||
|
pub boat_id: i64,
|
||||||
|
pub start_date: &'r str,
|
||||||
|
pub end_date: &'r str,
|
||||||
|
pub time_desc: &'r str,
|
||||||
|
pub usage: &'r str,
|
||||||
|
pub user_id_applicant: Option<i64>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[post("/", data = "<data>", rank = 2)]
|
||||||
|
async fn create<'r>(
|
||||||
|
db: &State<SqlitePool>,
|
||||||
|
data: Form<FormBoatReservationToAdd<'r>>,
|
||||||
|
user: DonauLinzUser,
|
||||||
|
) -> Flash<Redirect> {
|
||||||
|
let user_applicant: User = user.into();
|
||||||
|
let boat = Boat::find_by_id(db, data.boat_id as i32).await.unwrap();
|
||||||
|
let boatreservation_to_add = BoatReservationToAdd {
|
||||||
|
boat: &boat,
|
||||||
|
start_date: NaiveDate::parse_from_str(data.start_date, "%Y-%m-%d").unwrap(),
|
||||||
|
end_date: NaiveDate::parse_from_str(data.end_date, "%Y-%m-%d").unwrap(),
|
||||||
|
time_desc: data.time_desc,
|
||||||
|
usage: data.usage,
|
||||||
|
user_applicant: &user_applicant,
|
||||||
|
};
|
||||||
|
match BoatReservation::create(db, boatreservation_to_add).await {
|
||||||
|
Ok(_) => Flash::success(
|
||||||
|
Redirect::to("/boatreservation"),
|
||||||
|
"Reservierung erfolgreich hinzugefügt",
|
||||||
|
),
|
||||||
|
Err(e) => Flash::error(Redirect::to("/boatreservation"), format!("Fehler: {e}")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[post("/", data = "<data>")]
|
||||||
|
async fn create_from_kiosk<'r>(
|
||||||
|
db: &State<SqlitePool>,
|
||||||
|
data: Form<FormBoatReservationToAdd<'r>>,
|
||||||
|
_kiosk: KioskCookie,
|
||||||
|
) -> Flash<Redirect> {
|
||||||
|
let user_applicant: User = User::find_by_id(db, data.user_id_applicant.unwrap() as i32)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
let boat = Boat::find_by_id(db, data.boat_id as i32).await.unwrap();
|
||||||
|
let boatreservation_to_add = BoatReservationToAdd {
|
||||||
|
boat: &boat,
|
||||||
|
start_date: NaiveDate::parse_from_str(data.start_date, "%Y-%m-%d").unwrap(),
|
||||||
|
end_date: NaiveDate::parse_from_str(data.end_date, "%Y-%m-%d").unwrap(),
|
||||||
|
time_desc: data.time_desc,
|
||||||
|
usage: data.usage,
|
||||||
|
user_applicant: &user_applicant,
|
||||||
|
};
|
||||||
|
match BoatReservation::create(db, boatreservation_to_add).await {
|
||||||
|
Ok(_) => Flash::success(
|
||||||
|
Redirect::to("/boatreservation"),
|
||||||
|
"Reservierung erfolgreich hinzugefügt",
|
||||||
|
),
|
||||||
|
Err(e) => Flash::error(Redirect::to("/boatreservation"), format!("Fehler: {e}")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[get("/<reservation_id>/delete")]
|
||||||
|
async fn delete<'r>(
|
||||||
|
db: &State<SqlitePool>,
|
||||||
|
reservation_id: i32,
|
||||||
|
user: DonauLinzUser,
|
||||||
|
) -> Flash<Redirect> {
|
||||||
|
let reservation = BoatReservation::find_by_id(db, reservation_id)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
if user.id == reservation.user_id_applicant || user.has_role(db, "admin").await {
|
||||||
|
reservation.delete(db).await;
|
||||||
|
Flash::success(
|
||||||
|
Redirect::to("/boatreservation"),
|
||||||
|
"Reservierung erfolgreich gelöscht",
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
Flash::error(
|
||||||
|
Redirect::to("/boatreservation"),
|
||||||
|
format!("Nur der Reservierer darf die Reservierung löschen."),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn routes() -> Vec<Route> {
|
||||||
|
routes![index, index_kiosk, create, create_from_kiosk, delete]
|
||||||
|
}
|
@ -17,6 +17,7 @@ use tera::Context;
|
|||||||
|
|
||||||
use crate::model::{
|
use crate::model::{
|
||||||
boat::Boat,
|
boat::Boat,
|
||||||
|
boatreservation::BoatReservation,
|
||||||
log::Log,
|
log::Log,
|
||||||
logbook::{
|
logbook::{
|
||||||
LogToAdd, LogToFinalize, Logbook, LogbookCreateError, LogbookDeleteError,
|
LogToAdd, LogToFinalize, Logbook, LogbookCreateError, LogbookDeleteError,
|
||||||
@ -73,6 +74,7 @@ async fn index(
|
|||||||
}
|
}
|
||||||
|
|
||||||
context.insert("boats", &boats);
|
context.insert("boats", &boats);
|
||||||
|
context.insert("reservations", &BoatReservation::all_future(db).await);
|
||||||
context.insert("coxes", &coxes);
|
context.insert("coxes", &coxes);
|
||||||
context.insert("users", &users);
|
context.insert("users", &users);
|
||||||
context.insert("logtypes", &logtypes);
|
context.insert("logtypes", &logtypes);
|
||||||
@ -163,6 +165,7 @@ async fn kiosk(
|
|||||||
}
|
}
|
||||||
|
|
||||||
context.insert("boats", &boats);
|
context.insert("boats", &boats);
|
||||||
|
context.insert("reservations", &BoatReservation::all_future(db).await);
|
||||||
context.insert("coxes", &coxes);
|
context.insert("coxes", &coxes);
|
||||||
context.insert("users", &users);
|
context.insert("users", &users);
|
||||||
context.insert("logtypes", &logtypes);
|
context.insert("logtypes", &logtypes);
|
||||||
|
@ -26,6 +26,7 @@ pub(crate) mod admin;
|
|||||||
mod auth;
|
mod auth;
|
||||||
pub(crate) mod board;
|
pub(crate) mod board;
|
||||||
mod boatdamage;
|
mod boatdamage;
|
||||||
|
mod boatreservation;
|
||||||
mod cox;
|
mod cox;
|
||||||
mod ergo;
|
mod ergo;
|
||||||
mod log;
|
mod log;
|
||||||
@ -94,6 +95,7 @@ pub fn config(rocket: Rocket<Build>) -> Rocket<Build> {
|
|||||||
.mount("/notification", notification::routes())
|
.mount("/notification", notification::routes())
|
||||||
.mount("/stat", stat::routes())
|
.mount("/stat", stat::routes())
|
||||||
.mount("/boatdamage", boatdamage::routes())
|
.mount("/boatdamage", boatdamage::routes())
|
||||||
|
.mount("/boatreservation", boatreservation::routes())
|
||||||
.mount("/cox", cox::routes())
|
.mount("/cox", cox::routes())
|
||||||
.mount("/admin", admin::routes())
|
.mount("/admin", admin::routes())
|
||||||
.mount("/board", board::routes())
|
.mount("/board", board::routes())
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
<a href="/stat" class="px-2">Statistik</a>
|
<a href="/stat" class="px-2">Statistik</a>
|
||||||
<a href="/stat/boats" class="px-2">Bootsauswertung</a>
|
<a href="/stat/boats" class="px-2">Bootsauswertung</a>
|
||||||
<a href="/boatdamage" class="px-2">Bootsschaden</a>
|
<a href="/boatdamage" class="px-2">Bootsschaden</a>
|
||||||
|
<a href="/boatreservation" class="px-2">Bootsreservierung</a>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
82
templates/boatreservations.html.tera
Normal file
82
templates/boatreservations.html.tera
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
{% import "includes/macros" as macros %}
|
||||||
|
{% import "includes/forms/log" as log %}
|
||||||
|
{% extends "base" %}
|
||||||
|
{% block content %}
|
||||||
|
<div class="max-w-screen-lg w-full">
|
||||||
|
<h1 class="h1">Bootsreservierungen</h1>
|
||||||
|
<h2 class="text-md font-bold tracking-wide bg-primary-900 mt-3 p-3 text-white flex justify-between items-center rounded-md">
|
||||||
|
Neue Reservierung
|
||||||
|
<a href="#"
|
||||||
|
class="inline-flex justify-center rounded-md bg-primary-600 mx-1 px-3 py-2 text-sm font-semibold text-white hover:bg-primary-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-primary-600 cursor-pointer"
|
||||||
|
data-sidebar="true"
|
||||||
|
data-trigger="sidebar"
|
||||||
|
data-header="Neue Reservierung anlegen"
|
||||||
|
data-body="#new-reservation">
|
||||||
|
{% include "includes/plus-icon" %}
|
||||||
|
<span class="sr-only">Neue Reservierung eintragen</span>
|
||||||
|
</a>
|
||||||
|
</h2>
|
||||||
|
<div class="hidden">
|
||||||
|
<div id="new-reservation">
|
||||||
|
<form action="/boatreservation" method="post" class="grid gap-3">
|
||||||
|
{{ log::boat_select(only_ones=false, id='boat') }}
|
||||||
|
{% if not loggedin_user %}{{ macros::select(label='Reserviert von', data=user, name='user_id_applicant') }}{% endif %}
|
||||||
|
{{ macros::input(label='Beginn', name='start_date', type='date', required=true, wrapper_class='col-span-4') }}
|
||||||
|
{{ macros::input(label='Ende', name='end_date', type='date', required=true, wrapper_class='col-span-4') }}
|
||||||
|
{{ macros::input(label='Uhrzeit (zB ab 14:00 Uhr, ganztägig, ...)', name='time_desc', type='text', required=true, wrapper_class='col-span-4') }}
|
||||||
|
{{ macros::input(label='Zweck (Wanderfahrt, ...)', name='usage', type='text', required=true, wrapper_class='col-span-4') }}
|
||||||
|
<input type="submit"
|
||||||
|
class="btn btn-primary w-full col-span-4"
|
||||||
|
value="Reservierung eintragen" />
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="search-wrapper">
|
||||||
|
<label for="name" class="sr-only">Suche</label>
|
||||||
|
<input type="search"
|
||||||
|
name="name"
|
||||||
|
id="filter-js"
|
||||||
|
class="search-bar"
|
||||||
|
placeholder="Suchen nach Namen...">
|
||||||
|
</div>
|
||||||
|
<div id="filter-result-js" class="search-result"></div>
|
||||||
|
{% for reservation in boatreservations %}
|
||||||
|
<div data-filterable="true"
|
||||||
|
data-filter="{{ reservation.user_applicant.name }} {{ reservation.boat.name }}"
|
||||||
|
class="w-full border-t bg-white dark:bg-primary-900 text-black dark:text-white p-3">
|
||||||
|
<div class="w-full">
|
||||||
|
<strong>Boot:</strong>
|
||||||
|
{{ reservation.boat.name }}
|
||||||
|
<br />
|
||||||
|
<strong>Reservierung:</strong>
|
||||||
|
{{ reservation.user_applicant.name }}
|
||||||
|
<br />
|
||||||
|
<strong>Datum:</strong>
|
||||||
|
{{ reservation.start_date }}
|
||||||
|
{% if reservation.end_date != reservation.start_date %}
|
||||||
|
-
|
||||||
|
{{ reservation.end_date }}
|
||||||
|
{% endif %}
|
||||||
|
<br />
|
||||||
|
<strong>Uhrzeit:</strong>
|
||||||
|
{{ reservation.time_desc }}
|
||||||
|
<br />
|
||||||
|
<strong>Zweck:</strong>
|
||||||
|
{{ reservation.usage }}
|
||||||
|
{% if loggedin_user %}
|
||||||
|
{% if loggedin_user.id == reservation.user_applicant.id or "admin" in loggedin_user.roles %}
|
||||||
|
<div class="mt-3 text-right">
|
||||||
|
<a href="/boatreservation/{{ reservation.id }}/delete"
|
||||||
|
class="w-28 btn btn-alert"
|
||||||
|
onclick="return confirm('Willst du diese Reservierung wirklich löschen?');">
|
||||||
|
{% include "includes/delete-icon" %}
|
||||||
|
Löschen
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% endblock content %}
|
@ -104,6 +104,7 @@
|
|||||||
<strong class="block text-primary-900 dark:text-white">
|
<strong class="block text-primary-900 dark:text-white">
|
||||||
{{ log.departure | date(format="%H:%M") }}
|
{{ log.departure | date(format="%H:%M") }}
|
||||||
Uhr
|
Uhr
|
||||||
|
<small>({{ log.departure | date(format="%d.%m.%Y") }})</small>
|
||||||
</strong>
|
</strong>
|
||||||
<a href="#"
|
<a href="#"
|
||||||
data-sidebar="true"
|
data-sidebar="true"
|
||||||
@ -149,7 +150,8 @@
|
|||||||
{% set amount_rowers = log.rowers | length %}
|
{% set amount_rowers = log.rowers | length %}
|
||||||
{% set amount_guests = log.boat.amount_seats - amount_rowers %}
|
{% set amount_guests = log.boat.amount_seats - amount_rowers %}
|
||||||
{% if amount_guests > 0 %}
|
{% if amount_guests > 0 %}
|
||||||
Gäste <small class="text-gray-600">(ohne Account)</small>:
|
Gäste
|
||||||
|
<small class="text-gray-600">(ohne Account)</small>:
|
||||||
{{ amount_guests }}
|
{{ amount_guests }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if allowed_to_close and state == "on_water" %}
|
{% if allowed_to_close and state == "on_water" %}
|
||||||
@ -182,18 +184,23 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div {% if log.logtype %}class="mt-4 sm:mt-0"{% endif %}>
|
<div {% if log.logtype %}class="mt-4 sm:mt-0"{% endif %}>
|
||||||
<strong class="text-black dark:text-white">{{ log.boat.name }}</strong> <small class="text-gray-600 dark:text-gray-100">({{ log.shipmaster_user.name -}}
|
<strong class="text-black dark:text-white">{{ log.boat.name }}</strong>
|
||||||
|
<small class="text-gray-600 dark:text-gray-100">({{ log.shipmaster_user.name -}}
|
||||||
{% if log.shipmaster_only_steering %}
|
{% if log.shipmaster_only_steering %}
|
||||||
- handgesteuert
|
- handgesteuert
|
||||||
{% endif -%}
|
{% endif -%}
|
||||||
)</small>
|
)</small>
|
||||||
<small class="block text-gray-600 dark:text-gray-100">
|
<small class="block text-gray-600 dark:text-gray-100">
|
||||||
{% if state == "completed" and log.departure | date(format='%d.%m.%Y') == log.arrival | date(format='%d.%m.%Y') %}
|
{% if state == "completed" and log.departure | date(format='%d.%m.%Y') == log.arrival | date(format='%d.%m.%Y') %}
|
||||||
{{ log.departure | date(format='%d.%m.%Y') }} ({{ log.departure | date(format='%H:%M') }} - {{ log.arrival | date(format='%H:%M') }})
|
{{ log.departure | date(format='%d.%m.%Y') }}
|
||||||
|
({{ log.departure | date(format='%H:%M') }}
|
||||||
|
-
|
||||||
|
{{ log.arrival | date(format='%H:%M') }})
|
||||||
{% else %}
|
{% else %}
|
||||||
{{ log.departure | date(format='%d.%m.%Y (%H:%M)') }}
|
{{ log.departure | date(format='%d.%m.%Y (%H:%M)') }}
|
||||||
{% if state == "completed" %}
|
{% if state == "completed" %}
|
||||||
- {{ log.arrival | date(format='%d.%m.%Y (%H:%M)') }}
|
-
|
||||||
|
{{ log.arrival | date(format='%d.%m.%Y (%H:%M)') }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</small>
|
</small>
|
||||||
@ -205,7 +212,8 @@
|
|||||||
<div class="text-black dark:text-white">
|
<div class="text-black dark:text-white">
|
||||||
{{ log.destination }}
|
{{ log.destination }}
|
||||||
{% if state == "completed" %}
|
{% if state == "completed" %}
|
||||||
<small class="text-gray-600 dark:text-gray-100">({{ log.distance_in_km }} km)</small>
|
<small class="text-gray-600 dark:text-gray-100">({{ log.distance_in_km }}
|
||||||
|
km)</small>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if log.comments %}<span class="text-sm italic">- "{{ log.comments }}"</span>{% endif %}
|
{% if log.comments %}<span class="text-sm italic">- "{{ log.comments }}"</span>{% endif %}
|
||||||
</div>
|
</div>
|
||||||
@ -217,7 +225,9 @@
|
|||||||
{% if not loop.last or amount_guests > 0 and log.boat.name != 'Externes Boot' %},{% endif %}
|
{% if not loop.last or amount_guests > 0 and log.boat.name != 'Externes Boot' %},{% endif %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% if amount_guests > 0 and log.boat.name != 'Externes Boot' %}
|
{% if amount_guests > 0 and log.boat.name != 'Externes Boot' %}
|
||||||
Gäste <small class="text-gray-600 dark:text-gray-100">(ohne Account)</small>: {{ amount_guests }}
|
Gäste
|
||||||
|
<small class="text-gray-600 dark:text-gray-100">(ohne Account)</small>:
|
||||||
|
{{ amount_guests }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
@ -1,14 +1,26 @@
|
|||||||
{% macro boatreservation() %}
|
{% macro boatreservation() %}
|
||||||
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow grid gap-3"
|
<div class="bg-white dark:bg-primary-900 rounded-md shadow pb-2 mt-3">
|
||||||
style="margin-top: 10px">
|
<h2 class="h2">Reservierungen ({{ reservations | length }})</h2>
|
||||||
<h2 class="h2">Bootsreservierungen</h2>
|
<div class="grid grid-cols-1 gap-3 mb-3 w-full">
|
||||||
<div class="p2" style="margin-bottom: 10px;">
|
{% for reservation in reservations %}
|
||||||
<ul style="display: flex;
|
<div class="pt-2 px-3 border-t text-primary-900 dark:text-white">
|
||||||
justify-content: space-around;
|
<strong class="block">
|
||||||
padding: 0;
|
{{ reservation.start_date | date(format="%d.%m.%Y") }}
|
||||||
list-style: none">
|
{% if reservation.end_date != reservation.start_date %}
|
||||||
<li style="display: inline-block;">30.03. | Manuela Firmötz | Boot: Urfahr</li>
|
-
|
||||||
</ul>
|
{{ reservation.end_date | date(format="%d.%m.%Y") }}
|
||||||
|
{% endif %}
|
||||||
|
<small>({{ reservation.time_desc }})</small>
|
||||||
|
</strong>
|
||||||
|
<span class="block">
|
||||||
|
{{ reservation.boat.name }}
|
||||||
|
<small>({{ reservation.user_applicant.name }})</small>
|
||||||
|
</span>
|
||||||
|
<span class="text-sm italic">{{ reservation.usage }}</span>
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<p class="p-3 text-center text-black dark:text-white">Keine Reservierung</p>
|
||||||
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endmacro boatreservation %}
|
{% endmacro boatreservation %}
|
||||||
@ -70,6 +82,8 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
<a href="/boatdamage"
|
<a href="/boatdamage"
|
||||||
class="block w-100 py-2 hover:text-primary-600 border-t">Bootsschaden</a>
|
class="block w-100 py-2 hover:text-primary-600 border-t">Bootsschaden</a>
|
||||||
|
<a href="/boatreservation"
|
||||||
|
class="block w-100 py-2 hover:text-primary-600 border-t">Bootsreservierung</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -141,7 +155,7 @@
|
|||||||
{% if required %}required="required"{% endif %}>
|
{% if required %}required="required"{% endif %}>
|
||||||
{% if default %}<option selected value>{{ default }}</option>{% endif %}
|
{% if default %}<option selected value>{{ default }}</option>{% endif %}
|
||||||
{% for d in data %}
|
{% for d in data %}
|
||||||
<option value="{{ d.id }}" {% 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 }}}'{% endif %}>
|
<option value="{{ d.id }}" {% 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"] }}}' {% endif %}>
|
||||||
{% for displa in display -%}
|
{% for displa in display -%}
|
||||||
{%- if d[displa] -%}
|
{%- if d[displa] -%}
|
||||||
{{- d[displa] -}}
|
{{- d[displa] -}}
|
||||||
|
@ -54,19 +54,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</details>
|
</details>
|
||||||
</div>
|
</div>
|
||||||
{% if loggedin_user.weight and loggedin_user.sex and loggedin_user.dob %}
|
|
||||||
<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</h2>
|
|
||||||
<div class="p-3">
|
|
||||||
<ul class="list-none ms-2">
|
|
||||||
<li class="py-1">
|
|
||||||
<a href="/ergo" class="block w-100 py-2 hover:text-primary-600">Ergo</a>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
{% if "Donau Linz" in loggedin_user.roles and "Unterstützend" not in loggedin_user.roles and "Förderndes Mitglied" not in loggedin_user.roles %}
|
{% if "Donau Linz" in loggedin_user.roles and "Unterstützend" not in loggedin_user.roles and "Förderndes Mitglied" not in loggedin_user.roles %}
|
||||||
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
|
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
|
||||||
role="alert">
|
role="alert">
|
||||||
@ -90,6 +77,10 @@
|
|||||||
<li class="py-1">
|
<li class="py-1">
|
||||||
<a href="/boatdamage" class="block w-100 py-2 hover:text-primary-600">Bootsschaden</a>
|
<a href="/boatdamage" class="block w-100 py-2 hover:text-primary-600">Bootsschaden</a>
|
||||||
</li>
|
</li>
|
||||||
|
<li class="py-1">
|
||||||
|
<a href="/boatreservation"
|
||||||
|
class="block w-100 py-2 hover:text-primary-600">Bootsreservierung</a>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -162,6 +153,19 @@
|
|||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
{% if loggedin_user.weight and loggedin_user.sex and loggedin_user.dob %}
|
||||||
|
<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</h2>
|
||||||
|
<div class="p-3">
|
||||||
|
<ul class="list-none ms-2">
|
||||||
|
<li class="py-1">
|
||||||
|
<a href="/ergo" class="block w-100 py-2 hover:text-primary-600">Ergo</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
|
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5"
|
||||||
role="alert">
|
role="alert">
|
||||||
<h2 class="h2">Allgemein</h2>
|
<h2 class="h2">Allgemein</h2>
|
||||||
|
@ -9,27 +9,31 @@
|
|||||||
{{ macros::alert(message=flash.1, type=flash.0, class="sm:col-span-2 lg:col-span-3") }}
|
{{ macros::alert(message=flash.1, type=flash.0, class="sm:col-span-2 lg:col-span-3") }}
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{{ macros::boatreservation() }}
|
<div class="w-full grid lg:grid-cols-5 gap-3 mt-5">
|
||||||
<div class="w-full grid md:grid-cols-5 gap-3 mt-5">
|
<div>
|
||||||
<div class="bg-white dark:bg-primary-900 rounded-md hidden md:block shadow">
|
<div class="bg-white dark:bg-primary-900 rounded-md hidden md:block shadow">
|
||||||
<h2 class="h2">Boote</h2>
|
<h2 class="h2">Boote</h2>
|
||||||
<div>{{ log::show_boats(only_ones=false) }}</div>
|
<div>{{ log::show_boats(only_ones=false) }}</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="md:col-span-3 bg-white dark:bg-primary-900 rounded-md shadow">
|
<div class="lg:col-span-3 bg-white dark:bg-primary-900 rounded-md shadow">
|
||||||
<h2 class="h2">Neue Ausfahrt</h2>
|
<h2 class="h2">Neue Ausfahrt</h2>
|
||||||
<div class="p-3">{{ log::new(only_ones=false, shipmaster=-1) }}</div>
|
<div class="p-3">{{ log::new(only_ones=false, shipmaster=-1) }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="bg-white dark:bg-primary-900 rounded-md shadow">
|
<div>
|
||||||
<h2 class="h2">Am Wasser</h2>
|
<div class="bg-white dark:bg-primary-900 rounded-md shadow pb-2">
|
||||||
<div>
|
<h2 class="h2">Am Wasser</h2>
|
||||||
{% if on_water | length > 0 %}
|
<div>
|
||||||
{% for log in on_water %}
|
{% if on_water | length > 0 %}
|
||||||
{{ log::show(log=log, state="on_water", allowed_to_close=true, only_ones=false) }}
|
{% for log in on_water %}
|
||||||
{% endfor %}
|
{{ log::show(log=log, state="on_water", allowed_to_close=true, only_ones=false) }}
|
||||||
{% else %}
|
{% endfor %}
|
||||||
<p class="p-3 text-center text-black dark:text-white">Kein Boot am Wasser</p>
|
{% else %}
|
||||||
{% endif %}
|
<p class="p-3 text-center text-black dark:text-white">Kein Boot am Wasser</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{{ macros::boatreservation() }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -4,31 +4,35 @@
|
|||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
<h1 class="h1">Logbuch</h1>
|
<h1 class="h1">Logbuch</h1>
|
||||||
{{ macros::boatreservation() }}
|
<div class="w-full grid lg:grid-cols-5 gap-3 mt-5">
|
||||||
<div class="w-full grid md:grid-cols-5 gap-3 mt-5">
|
<div>
|
||||||
<div class="bg-white dark:bg-primary-900 rounded-md hidden md:block shadow">
|
<div class="bg-white dark:bg-primary-900 rounded-md hidden md:block shadow">
|
||||||
<h2 class="h2">Boote</h2>
|
<h2 class="h2">Boote</h2>
|
||||||
<div>{{ log::show_boats(only_ones=false) }}</div>
|
<div>{{ log::show_boats(only_ones=false) }}</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="md:col-span-3 bg-white dark:bg-primary-900 rounded-md shadow">
|
<div class="lg:col-span-3 bg-white dark:bg-primary-900 rounded-md shadow">
|
||||||
<h2 class="h2">Neue Ausfahrt</h2>
|
<h2 class="h2">Neue Ausfahrt</h2>
|
||||||
<div class="p-3">{{ log::new(shipmaster=loggedin_user.id) }}</div>
|
<div class="p-3">{{ log::new(shipmaster=loggedin_user.id) }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="bg-white dark:bg-primary-900 rounded-md shadow">
|
<div>
|
||||||
<h2 class="h2">Am Wasser</h2>
|
<div class="bg-white dark:bg-primary-900 rounded-md shadow pb-2">
|
||||||
{% if on_water | length > 0 %}
|
<h2 class="h2">Am Wasser</h2>
|
||||||
{% for log in on_water %}
|
{% if on_water | length > 0 %}
|
||||||
{% if log.shipmaster == loggedin_user.id %}
|
{% for log in on_water %}
|
||||||
{{ log::show(log=log, state="on_water", allowed_to_close=true, only_ones="cox" not in loggedin_user.roles) }}
|
{% if log.shipmaster == loggedin_user.id %}
|
||||||
{% elif "Vorstand" in loggedin_user.roles %}
|
{{ log::show(log=log, state="on_water", allowed_to_close=true, only_ones="cox" not in loggedin_user.roles) }}
|
||||||
{{ log::show(log=log, state="on_water", allowed_to_close=true, only_ones="cox" not in loggedin_user.roles) }}
|
{% elif "Vorstand" in loggedin_user.roles %}
|
||||||
{% else %}
|
{{ log::show(log=log, state="on_water", allowed_to_close=true, only_ones="cox" not in loggedin_user.roles) }}
|
||||||
{{ log::show(log=log, state="on_water", only_ones=true) }}
|
{% else %}
|
||||||
{% endif %}
|
{{ log::show(log=log, state="on_water", only_ones=true) }}
|
||||||
{% endfor %}
|
{% endif %}
|
||||||
{% else %}
|
{% endfor %}
|
||||||
<p class="p-3 text-center text-black dark:text-white">Kein Boot am Wasser</p>
|
{% else %}
|
||||||
{% endif %}
|
<p class="p-3 text-center text-black dark:text-white">Kein Boot am Wasser</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{{ macros::boatreservation() }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
Loading…
Reference in New Issue
Block a user