2025-09-02 14:00:19 -04:00
|
|
|
use dioxus::prelude::*;
|
2025-09-06 15:05:58 -04:00
|
|
|
use tavern::{Adventurer, Legend, Lore, Tale};
|
2025-09-02 14:00:19 -04:00
|
|
|
|
2025-09-08 10:01:02 -04:00
|
|
|
use crate::page::Page;
|
2025-09-06 15:05:58 -04:00
|
|
|
use crate::server::*;
|
2025-09-02 14:00:19 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const AUTHOR_CSS: Asset = asset!("/assets/css/blog_author.css");
|
|
|
|
|
const LIST_CSS: Asset = asset!("/assets/css/blog_list.css");
|
|
|
|
|
const NAV_CSS: Asset = asset!("/assets/css/blog_nav.css");
|
|
|
|
|
const POST_CSS: Asset = asset!("/assets/css/blog_post.css");
|
|
|
|
|
const TAG_CSS: Asset = asset!("/assets/css/blog_tag.css");
|
|
|
|
|
|
|
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
|
|
|
|
|
fn convert_tag<S>(text: S) -> (String, String)
|
|
|
|
|
where S: AsRef<str>
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
if let Some((first, second)) = text.as_ref().split_once('-')
|
|
|
|
|
{
|
|
|
|
|
(first.to_owned(), second.to_owned())
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
(text.as_ref().to_owned(), "none".to_owned())
|
|
|
|
|
}
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
fn strip_tag(text: &str) -> &str
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
if let Some((first, _)) = text.split_once('-')
|
|
|
|
|
{
|
|
|
|
|
first
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
text
|
|
|
|
|
}
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[component]
|
2025-09-08 18:39:30 -04:00
|
|
|
pub fn TagItem(tag: String, style: String) -> Element
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-02 14:00:19 -04:00
|
|
|
li
|
|
|
|
|
{
|
|
|
|
|
class: "tag_item",
|
2025-09-08 18:39:30 -04:00
|
|
|
a { class: "{style}", "{tag}" }
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[component]
|
|
|
|
|
pub fn TagList(children: Element) -> Element
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-02 14:00:19 -04:00
|
|
|
document::Link { rel: "stylesheet", href: TAG_CSS }
|
|
|
|
|
|
|
|
|
|
h5
|
|
|
|
|
{
|
|
|
|
|
class: "blog_tag_style",
|
|
|
|
|
ul
|
|
|
|
|
{
|
|
|
|
|
class: "tag_list",
|
|
|
|
|
{children}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[component]
|
|
|
|
|
pub fn BlogAuthor() -> Element
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-02 14:00:19 -04:00
|
|
|
document::Link { rel: "stylesheet", href: AUTHOR_CSS }
|
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
{
|
|
|
|
|
class: "blog_author_style",
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[component]
|
2025-09-08 18:39:30 -04:00
|
|
|
pub fn BlogItem(title: String, slug: String, author: String, summary: String,
|
|
|
|
|
tags: Vec<String>)
|
|
|
|
|
-> Element
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
let tags: Vec<(String, String)> =
|
|
|
|
|
tags.iter().map(|t| convert_tag(t)).collect();
|
|
|
|
|
|
|
|
|
|
rsx! {
|
2025-09-02 14:00:19 -04:00
|
|
|
li
|
|
|
|
|
{
|
|
|
|
|
class: "blog_item",
|
|
|
|
|
Link
|
|
|
|
|
{
|
|
|
|
|
to: Page::Post { slug: slug },
|
|
|
|
|
h1 { "{title}" }
|
|
|
|
|
}
|
|
|
|
|
TagList
|
|
|
|
|
{
|
|
|
|
|
for (tag, style) in tags
|
|
|
|
|
{
|
|
|
|
|
TagItem
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
tag: tag.clone(),
|
|
|
|
|
style: style.clone()
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
h4 { "Author: {author}" }
|
|
|
|
|
p { "{summary}" }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[component]
|
2025-09-08 10:01:02 -04:00
|
|
|
pub fn BlogList(tags: Vec<String>, children: Element) -> Element
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
let summaries = use_server_future(move || {
|
2025-09-08 10:01:02 -04:00
|
|
|
let categories = tags.clone();
|
2025-09-08 18:39:30 -04:00
|
|
|
async move { get_blog_list(categories).await }
|
2025-09-08 10:01:02 -04:00
|
|
|
})?;
|
|
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-02 14:00:19 -04:00
|
|
|
document::Link { rel: "stylesheet", href: LIST_CSS }
|
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
{
|
|
|
|
|
class: "blog_list_style",
|
|
|
|
|
ul
|
|
|
|
|
{
|
|
|
|
|
class: "blog_list",
|
2025-09-08 10:01:02 -04:00
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
match &*summaries.read()
|
|
|
|
|
{
|
|
|
|
|
Some(Ok(lores)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
for lore in lores
|
|
|
|
|
{
|
|
|
|
|
BlogItem
|
|
|
|
|
{
|
|
|
|
|
title: lore.title.clone(),
|
|
|
|
|
slug: lore.slug.clone(),
|
|
|
|
|
author: lore.author.clone(),
|
|
|
|
|
summary: lore.summary.clone(),
|
|
|
|
|
tags: lore.tags.clone()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Some(Err(e)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
p { "Unable to show post header." }
|
|
|
|
|
p { "{e}" }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
None =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
p { "Loading..." }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-02 14:00:19 -04:00
|
|
|
{children}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[component]
|
|
|
|
|
pub fn BlogNav() -> Element
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
let tags = use_server_future(move || async move { get_tags().await })?;
|
2025-09-08 10:01:02 -04:00
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-02 14:00:19 -04:00
|
|
|
document::Link { rel: "stylesheet", href: NAV_CSS }
|
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
{
|
|
|
|
|
class: "blog_nav_style",
|
|
|
|
|
div
|
|
|
|
|
{
|
|
|
|
|
class: "tag_style",
|
|
|
|
|
h3 { "Categories" }
|
|
|
|
|
ul
|
|
|
|
|
{
|
|
|
|
|
class: "tag_list",
|
2025-09-08 18:39:30 -04:00
|
|
|
match &*tags.read()
|
|
|
|
|
{
|
|
|
|
|
Some(Ok(categories)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
for tag in categories
|
|
|
|
|
{
|
2025-09-02 14:00:19 -04:00
|
|
|
li
|
|
|
|
|
{
|
|
|
|
|
class: "tag_item",
|
2025-09-08 18:39:30 -04:00
|
|
|
a { href: "/blog/{strip_tag(tag)}", "{strip_tag(tag)}" }
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
2025-09-08 18:39:30 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Some(Err(e)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
p { "Unable to show desired post." }
|
|
|
|
|
p { "{e}" }
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
2025-09-08 18:39:30 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
None =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
p { "Loading..." }
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-09-08 18:39:30 -04:00
|
|
|
}
|
|
|
|
|
}
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[component]
|
2025-09-08 18:39:30 -04:00
|
|
|
pub fn PostHeaderAuthor(name: String, handle: String, profile_link: String)
|
|
|
|
|
-> Element
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-06 15:05:58 -04:00
|
|
|
h4 { "Author: ", a { href: "{profile_link}", "{name} @{handle}" } }
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-09-02 14:00:19 -04:00
|
|
|
|
2025-09-06 15:05:58 -04:00
|
|
|
#[component]
|
|
|
|
|
pub fn PostHeader(title: String, author: String, tags: Vec<String>) -> Element
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
let author = use_server_future(move || {
|
2025-09-07 19:27:26 -04:00
|
|
|
let target_author = author.clone();
|
2025-09-08 18:39:30 -04:00
|
|
|
async move { get_author(target_author).await }
|
2025-09-07 19:27:26 -04:00
|
|
|
})?;
|
|
|
|
|
|
2025-09-06 15:05:58 -04:00
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
let converted_tags: Vec<(String, String)> =
|
|
|
|
|
tags.iter().map(|t| convert_tag(t)).collect();
|
|
|
|
|
|
|
|
|
|
rsx! {
|
2025-09-06 15:05:58 -04:00
|
|
|
h1 { "{title}" }
|
|
|
|
|
TagList
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
for (tag, style) in converted_tags
|
2025-09-02 14:00:19 -04:00
|
|
|
{
|
|
|
|
|
TagItem
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
tag: tag.clone(),
|
|
|
|
|
style: style.clone()
|
2025-09-02 14:00:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
2025-09-06 15:05:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
match &*author.read()
|
|
|
|
|
{
|
|
|
|
|
Some(Ok(author)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
PostHeaderAuthor
|
|
|
|
|
{
|
|
|
|
|
name: author.name.clone(),
|
|
|
|
|
handle: author.handle.clone(),
|
|
|
|
|
profile_link: author.legend.profile.clone()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Some(Err(e)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
p { "Unable to show post header." }
|
|
|
|
|
p { "{e}" }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
None =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
p { "Loading..." }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[component]
|
|
|
|
|
pub fn BlogPost(slug: String, children: Element) -> Element
|
|
|
|
|
{
|
2025-09-08 18:39:30 -04:00
|
|
|
let post = use_server_future(move || {
|
2025-09-06 22:48:58 -04:00
|
|
|
let url_slug = slug.clone();
|
2025-09-08 18:39:30 -04:00
|
|
|
async move { get_blog_post(url_slug).await }
|
2025-09-06 22:48:58 -04:00
|
|
|
})?;
|
2025-09-06 15:05:58 -04:00
|
|
|
|
2025-09-08 18:39:30 -04:00
|
|
|
rsx! {
|
2025-09-06 15:05:58 -04:00
|
|
|
document::Link { rel: "stylesheet", href: POST_CSS }
|
|
|
|
|
|
|
|
|
|
article
|
|
|
|
|
{
|
|
|
|
|
class: "blog_post_style",
|
|
|
|
|
|
|
|
|
|
match &*post.read()
|
|
|
|
|
{
|
|
|
|
|
Some(Ok(post)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
PostHeader
|
|
|
|
|
{
|
|
|
|
|
title: post.lore.title.clone(),
|
|
|
|
|
author: post.lore.author.clone(),
|
|
|
|
|
tags: post.lore.tags.clone()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
div { dangerous_inner_html: "{post.story}" }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Some(Err(e)) =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
p { "Unable to show desired post." }
|
|
|
|
|
p { "{e}" }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
None =>
|
|
|
|
|
{
|
|
|
|
|
rsx!
|
|
|
|
|
{
|
|
|
|
|
p { "Loading..." }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-02 14:00:19 -04:00
|
|
|
{children}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|