2025-01-17 19:35:35 +01:00
|
|
|
use based::ui::primitives::space::Fraction;
|
|
|
|
use based::ui::{prelude::*, AttrExtendable};
|
2024-12-18 20:09:07 +01:00
|
|
|
use based::{
|
|
|
|
auth::MaybeUser,
|
|
|
|
format::format_date,
|
|
|
|
request::{RequestContext, StringResponse},
|
2024-12-14 00:24:10 +01:00
|
|
|
};
|
2025-01-17 19:35:35 +01:00
|
|
|
use maud::{html, PreEscaped, Render};
|
2024-12-18 20:09:07 +01:00
|
|
|
use rocket::{get, State};
|
2024-12-14 00:24:10 +01:00
|
|
|
|
2025-01-17 19:35:35 +01:00
|
|
|
use crate::library::Video;
|
|
|
|
use crate::yt_meta::YouTubeMeta;
|
2024-12-22 20:19:52 +01:00
|
|
|
use crate::{
|
|
|
|
library::{history::VideoHistory, Library},
|
|
|
|
pages::components::video_element_wide,
|
|
|
|
};
|
2024-12-14 00:24:10 +01:00
|
|
|
|
2024-12-18 19:24:38 +01:00
|
|
|
use super::components::render_page;
|
2024-12-14 00:24:10 +01:00
|
|
|
|
|
|
|
#[get("/watch?<v>")]
|
|
|
|
pub async fn watch_page(
|
2024-12-18 19:24:38 +01:00
|
|
|
ctx: RequestContext,
|
2024-12-14 00:24:10 +01:00
|
|
|
library: &State<Library>,
|
|
|
|
v: String,
|
2024-12-18 20:09:07 +01:00
|
|
|
user: MaybeUser,
|
|
|
|
) -> StringResponse {
|
2024-12-14 00:24:10 +01:00
|
|
|
let video = if let Some(video) = library.get_video_by_id(&v).await {
|
|
|
|
video
|
|
|
|
} else {
|
|
|
|
// TODO : Error handling
|
|
|
|
library.get_video_by_youtube_id(&v).await.unwrap()
|
|
|
|
};
|
|
|
|
|
2024-12-22 20:19:52 +01:00
|
|
|
if let Some(user) = user.user() {
|
|
|
|
user.insert_history(video.id).await;
|
|
|
|
}
|
|
|
|
|
2025-01-17 19:35:35 +01:00
|
|
|
let youtube_meta = video.youtube_meta().await;
|
|
|
|
let rec = build_rec(&library, &video).await;
|
2024-12-14 23:33:47 +01:00
|
|
|
|
2025-01-17 19:35:35 +01:00
|
|
|
let content = Container(
|
|
|
|
Margin(
|
|
|
|
Screen::large(Flex(Nothing()).direction(Direction::Row)).on(
|
|
|
|
Flex(
|
|
|
|
Div().vanish()
|
|
|
|
.push(
|
|
|
|
Margin(
|
|
|
|
Screen::large(Width(Fraction::_10on12, Nothing())).on(
|
|
|
|
Div().push(
|
|
|
|
Context(Aspect::video(
|
2025-01-21 14:12:43 +01:00
|
|
|
Background(
|
2025-01-17 19:35:35 +01:00
|
|
|
Rounded(
|
|
|
|
html! {
|
|
|
|
video
|
|
|
|
controls
|
|
|
|
autoplay
|
2025-01-21 14:12:43 +01:00
|
|
|
class="w-full h-full rounded-lg" {
|
2025-01-17 19:35:35 +01:00
|
|
|
source src=(format!("/video/raw?v={}", video.id)) type="video/mp4" {
|
|
|
|
"Your browser does not support the video"
|
|
|
|
};
|
2024-12-14 23:33:47 +01:00
|
|
|
};
|
2025-01-17 19:35:35 +01:00
|
|
|
}
|
|
|
|
).size(Size::Large)
|
2025-01-21 14:12:43 +01:00
|
|
|
).color(Colors::Black)
|
2025-01-17 19:35:35 +01:00
|
|
|
))
|
|
|
|
).push(
|
|
|
|
Context(Margin(Padding(
|
2025-01-21 14:12:43 +01:00
|
|
|
Background(
|
2025-01-17 19:35:35 +01:00
|
|
|
Rounded(
|
|
|
|
Shadow::large(
|
|
|
|
Div()
|
|
|
|
.push(
|
|
|
|
Text(&video.title)._2xl().semibold()
|
|
|
|
)
|
|
|
|
.push_some(youtube_meta.as_ref(), |meta: &_| {
|
2025-01-21 14:12:43 +01:00
|
|
|
Div()
|
2025-01-17 19:35:35 +01:00
|
|
|
.push(
|
|
|
|
Margin(Flex(
|
|
|
|
Div().vanish().push(
|
|
|
|
Margin(Text(&format_date(&meta.upload_date)).color(&Gray::_300)).bottom(ScreenValue::_4)
|
|
|
|
).push(
|
|
|
|
Margin(Text(&format!("{} Views ﹣ {}", meta.views, format_date(&meta.upload_date))).color(&Gray::_300)).bottom(ScreenValue::_4)
|
|
|
|
)
|
|
|
|
).justify(Justify::Between).group()).top(ScreenValue::_2)
|
|
|
|
)
|
|
|
|
.push(
|
|
|
|
Link(&format!("https://www.youtube.com/watch?v={}", meta.id),
|
|
|
|
Text("Watch on YouTube").color(&Red::_400)
|
|
|
|
)
|
|
|
|
).push(
|
|
|
|
Margin(Text(&meta.description).bold().color(&Gray::_300).wrap(TextWrap::Pretty).whitespace(TextWhitespace::BreakSpaces)).bottom(ScreenValue::_2).top(ScreenValue::_2)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)).size(Size::Large)
|
2025-01-21 14:12:43 +01:00
|
|
|
).color(Stone::_900)
|
2025-01-17 19:35:35 +01:00
|
|
|
).all(ScreenValue::_4)).top(ScreenValue::_8))
|
|
|
|
)
|
|
|
|
)
|
|
|
|
).top(ScreenValue::_10)
|
|
|
|
).push(
|
|
|
|
rec
|
|
|
|
)
|
|
|
|
).direction(Direction::Column).gap(ScreenValue::_6))
|
|
|
|
).x(ScreenValue::auto).top(ScreenValue::_6)
|
|
|
|
).render();
|
2024-12-14 00:24:10 +01:00
|
|
|
|
2024-12-18 19:24:38 +01:00
|
|
|
render_page(
|
|
|
|
ctx,
|
|
|
|
content,
|
|
|
|
&format!("{} - WatchDogs", video.title),
|
2024-12-18 20:09:07 +01:00
|
|
|
user.into(),
|
2024-12-18 19:24:38 +01:00
|
|
|
)
|
|
|
|
.await
|
2024-12-14 00:24:10 +01:00
|
|
|
}
|
2025-01-17 19:35:35 +01:00
|
|
|
|
|
|
|
pub async fn build_rec(library: &Library, video: &Video) -> PreEscaped<String> {
|
|
|
|
let videos = library.get_directory_videos(&video.directory).await;
|
|
|
|
|
|
|
|
let video_elements = html! {
|
|
|
|
@for video in videos {
|
|
|
|
(video_element_wide(&video).await);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
Margin(Width(
|
|
|
|
Fraction::_1on3,
|
|
|
|
Div()
|
|
|
|
.id("recommendations")
|
|
|
|
.push(
|
|
|
|
Margin(
|
|
|
|
Paragraph(Context(
|
|
|
|
SpaceBetween(
|
|
|
|
Flex(Div().vanish().push(Span("In ")).push(Link(
|
|
|
|
&format!("/d/{}", video.directory),
|
|
|
|
Text(&video.directory).color(&Blue::_500),
|
|
|
|
)))
|
|
|
|
.group()
|
|
|
|
.justify(Justify::Center),
|
|
|
|
)
|
|
|
|
.x(ScreenValue::_2),
|
|
|
|
))
|
|
|
|
.align(TextAlignment::Center)
|
|
|
|
._4xl()
|
|
|
|
.extrabold(),
|
|
|
|
)
|
|
|
|
.bottom(ScreenValue::_2),
|
|
|
|
)
|
|
|
|
.push(video_elements),
|
|
|
|
))
|
|
|
|
.top(ScreenValue::_8)
|
|
|
|
.render()
|
|
|
|
}
|