dioxus_docs_kit/components/blog/
blog_post.rs1use dioxus::prelude::*;
2use dioxus_free_icons::Icon;
3use dioxus_free_icons::icons::ld_icons::LdClock;
4use dioxus_mdx::{DocContent, DocTableOfContents, extract_headers};
5
6use crate::BlogContext;
7use crate::blog::registry::BlogRegistry;
8
9use super::author_info::AuthorInfo;
10use super::blog_meta::BlogPostMeta;
11use super::post_nav::BlogPostNav;
12use super::progress_bar::ReadingProgressBar;
13use super::related_posts::RelatedPosts;
14
15#[component]
17pub fn BlogPostView(slug: String) -> Element {
18 let registry = use_context::<&'static BlogRegistry>();
19 let ctx = use_context::<BlogContext>();
20
21 let post = match registry.get_post(&slug) {
22 Some(p) => p,
23 None => {
24 let base = ctx.base_path.clone();
25 return rsx! {
26 div { class: "max-w-4xl mx-auto px-4 py-12",
27 div { class: "text-center",
28 h1 { class: "text-4xl font-bold mb-4", "404" }
29 p { class: "text-base-content/70 mb-8",
30 "Post not found: {slug}"
31 }
32 Link {
33 to: NavigationTarget::Internal(base),
34 class: "btn btn-primary",
35 "Back to Blog"
36 }
37 }
38 }
39 };
40 }
41 };
42
43 let date_display = registry.format_date(&post.frontmatter.date);
44 let headers = extract_headers(&post.raw_markdown);
45
46 rsx! {
47 ReadingProgressBar {}
48 if let Some(ref site_url) = ctx.site_url {
49 BlogPostMeta { slug: slug.clone(), site_url: site_url.clone() }
50 }
51 div { class: "flex max-w-6xl mx-auto",
52 main { class: "flex-1 min-w-0 px-4 py-12 lg:px-12",
53 article { class: "max-w-3xl mx-auto",
54 if let Some(ref cover) = post.frontmatter.cover_image {
55 div { class: "mb-8 rounded-xl overflow-hidden",
56 img {
57 src: "{cover}",
58 alt: "{post.frontmatter.title}",
59 class: "w-full",
60 }
61 }
62 }
63
64 header { class: "mb-8 pb-8 border-b border-base-300",
65 if !post.frontmatter.tags.is_empty() {
66 div { class: "flex flex-wrap gap-1.5 mb-4",
67 for tag in post.frontmatter.tags.iter() {
68 span { class: "badge badge-sm badge-outline badge-primary font-medium",
69 "{tag}"
70 }
71 }
72 }
73 }
74
75 h1 { class: "text-4xl font-bold tracking-tight mb-4",
76 "{post.frontmatter.title}"
77 }
78
79 if let Some(ref desc) = post.frontmatter.description {
80 p { class: "text-lg text-base-content/60 mb-6",
81 "{desc}"
82 }
83 }
84
85 div { class: "flex items-center gap-4 text-sm text-base-content/60",
86 AuthorInfo { author_id: post.frontmatter.author.clone() }
87 span { class: "text-base-content/30", "|" }
88 span { "{date_display}" }
89 span { class: "text-base-content/30", "|" }
90 div { class: "flex items-center gap-1",
91 Icon { class: "size-3.5", icon: LdClock }
92 span { "{post.reading_time_minutes} min read" }
93 }
94 }
95 }
96
97 div { class: "prose prose-base max-w-none
98 prose-headings:scroll-mt-20
99 prose-h2:text-2xl prose-h2:font-semibold prose-h2:mt-10 prose-h2:mb-4
100 prose-h3:text-xl prose-h3:font-medium prose-h3:mt-8 prose-h3:mb-3
101 prose-p:text-base-content/80 prose-p:leading-relaxed
102 prose-a:text-primary prose-a:no-underline hover:prose-a:underline
103 prose-code:bg-base-200 prose-code:px-1.5 prose-code:py-0.5 prose-code:rounded prose-code:text-sm
104 prose-pre:bg-base-200 prose-pre:border prose-pre:border-base-300",
105 DocContent { nodes: post.content.clone() }
106 }
107
108 RelatedPosts { slug: slug.clone() }
109 BlogPostNav { current_slug: slug.clone() }
110 }
111 }
112
113 if !headers.is_empty() {
114 aside { class: "w-56 shrink-0 hidden xl:block",
115 div { class: "sticky top-20 p-6",
116 DocTableOfContents { headers }
117 }
118 }
119 }
120 }
121 }
122}