1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
|
/*
This file is part of jellything (https://codeberg.org/metamuffin/jellything)
which is licensed under the GNU Affero General Public License (version 3); see /COPYING.
Copyright (C) 2024 metamuffin <metamuffin.org>
*/
use crate::{
routes::ui::{
account::{
rocket_uri_macro_r_account_login, rocket_uri_macro_r_account_logout,
rocket_uri_macro_r_account_register, session::Session,
settings::rocket_uri_macro_r_account_settings,
},
admin::rocket_uri_macro_r_admin_dashboard,
browser::rocket_uri_macro_r_all_items,
node::rocket_uri_macro_r_library_node,
search::rocket_uri_macro_r_search,
},
uri,
};
use futures::executor::block_on;
use jellybase::CONF;
use jellycommon::user::Theme;
use markup::{DynRender, Render};
use rocket::{
http::ContentType,
response::{self, Responder},
Request, Response,
};
use std::{io::Cursor, sync::LazyLock};
static LOGO_ENABLED: LazyLock<bool> = LazyLock::new(|| CONF.asset_path.join("logo.svg").exists());
markup::define! {
Layout<'a, Main: Render>(title: String, main: Main, class: &'a str, session: Option<Session>) {
@markup::doctype()
html {
head {
title { @title " - " @CONF.brand }
meta[name="viewport", content="width=device-width, initial-scale=1.0"];
link[rel="stylesheet", href="/assets/style.css"];
script[src="/assets/bundle.js"] {}
}
body[class=class] {
nav {
h1 { a[href="/"] { @if *LOGO_ENABLED { img.logo[src="/assets/logo.svg"]; } else { @CONF.brand } } } " "
@if let Some(_) = session {
a.library[href=uri!(r_library_node("library"))] { "My Library" } " "
a.library[href=uri!(r_all_items())] { "All Items" } " "
a.library[href=uri!(r_search(None::<&'static str>, None::<usize>))] { "Search" } " "
}
div.account {
@if let Some(session) = session {
span { "Logged in as " } span.username { @session.user.display_name } " "
@if session.user.admin {
a.admin.hybrid_button[href=uri!(r_admin_dashboard())] { p {"Administration"} } " "
}
a.settings.hybrid_button[href=uri!(r_account_settings())] { p {"Settings"} } " "
a.logout.hybrid_button[href=uri!(r_account_logout())] { p {"Log out"} }
} else {
a.register.hybrid_button[href=uri!(r_account_register())] { p {"Register"} } " "
a.login.hybrid_button[href=uri!(r_account_login())] { p {"Log in"} }
}
}
}
#main { @main }
footer {
p { @CONF.brand " - " @CONF.slogan " | powered by Jellything" }
}
}
}
}
FlashDisplay(flash: Option<Result<String, String>>) {
@if let Some(flash) = &flash {
@match flash {
Ok(mesg) => { section.message { p.success { @mesg } } }
Err(err) => { section.message { p.error { @err } } }
}
}
}
}
pub type DynLayoutPage<'a> = LayoutPage<markup::DynRender<'a>>;
pub struct LayoutPage<T> {
pub title: String,
pub class: Option<&'static str>,
pub content: T,
}
impl Default for LayoutPage<DynRender<'_>> {
fn default() -> Self {
Self {
class: None,
content: markup::new!(),
title: String::new(),
}
}
}
impl<'r, Main: Render> Responder<'r, 'static> for LayoutPage<Main> {
fn respond_to(self, req: &'r Request<'_>) -> response::Result<'static> {
// TODO blocking the event loop here. it seems like there is no other way to
// TODO offload this, since the guard references `req` which has a lifetime.
// TODO therefore we just block. that is fine since the database is somewhat fast.
let session = block_on(req.guard::<Option<Session>>()).unwrap();
let mut out = String::new();
Layout {
main: self.content,
title: self.title,
class: &format!(
"{} theme-{:?}",
self.class.unwrap_or(""),
session
.as_ref()
.map(|s| s.user.theme)
.unwrap_or(Theme::Dark)
),
session,
}
.render(&mut out)
.unwrap();
Response::build()
.header(ContentType::HTML)
.streamed_body(Cursor::new(out))
.ok()
}
}
|