use ellipse::Ellipse;
pub static POSTS_PER_PAGE: usize = 20;
/// Return a pretty location specifier from a geo: URI.
fn decode_geo_uri(uri: &str) -> String {
if let Some(part) = uri.split(':').collect::<Vec<_>>().get(1) {
if let Some(part) = part.split(';').next() {
let mut parts = part.split(',');
let lat = parts.next().unwrap();
let lon = parts.next().unwrap();
// TODO - format them as proper latitude and longitude
format!("{}, {}", lat, lon)
} else {
uri.to_string()
}
} else {
uri.to_string()
}
}
markup::define! {
Entry<'a>(post: &'a serde_json::Value, from_feed: bool) {
@if post.pointer("/properties/like-of").is_none() && post.pointer("/properties/bookmark-of").is_none() {
@FullEntry { post, from_feed: *from_feed }
} else {
// Show a mini-post.
@MiniEntry { post }
}
}
MiniEntry<'a>(post: &'a serde_json::Value) {
article."h-entry mini-entry" {
@if let Some(author) = post["properties"]["author"][0].as_object() {
span."mini-h-card"."u-author" {
a."u-author"[href=author["properties"]["uid"][0].as_str().unwrap()] {
@if let Some(photo) = author["properties"]["photo"][0].as_str() {
img[src=photo, loading="lazy"];
} else if author["properties"]["photo"][0].is_object() {
img[
src=author["properties"]["photo"][0]["value"].as_str().unwrap(),
alt=author["properties"]["photo"][0]["alt"].as_str().unwrap(),
loading="lazy"
];
}
@author["properties"]["name"][0].as_str().unwrap()
}
}
@if post["properties"].as_object().unwrap().contains_key("like-of") {
" "
span."like-icon"["aria-label"="liked"] {
span."like-icon-label"["aria-hidden"="true"] {
"❤️"
}
}
" "
@if let Some(likeof) = post["properties"]["like-of"][0].as_str() {
a."u-like-of"[href=likeof] { @likeof }
} else if let Some(likeof) = post["properties"]["like-of"][0].as_object() {
a."u-like-of"[href=likeof["properties"]["url"][0].as_str().unwrap()] {
@likeof["properties"]["name"][0]
.as_str()
.and_then(|s| if s.trim().is_empty() { None } else { Some(s) })
.unwrap_or_else(|| likeof["properties"]["url"][0].as_str().unwrap())
}
}
} else if post["properties"].as_object().unwrap().contains_key("bookmark-of") {
" "
span."bookmark-icon"["aria-label"="bookmarked"] {
span."bookmark-icon-label"["aria-hidden"="true"] {
"🔖"
}
}
" "
@if let Some(bookmarkof) = post["properties"]["bookmark-of"][0].as_str() {
a."u-bookmark-of"[href=bookmarkof] { @bookmarkof }
} else if let Some(bookmarkof) = post["properties"]["bookmark-of"][0].as_object() {
a."u-bookmark-of"[href=bookmarkof["properties"]["url"][0].as_str().unwrap()] {
@bookmarkof["properties"]["name"][0]
.as_str()
.and_then(|s| if s.trim().is_empty() { None } else { Some(s) })
.unwrap_or_else(|| bookmarkof["properties"]["url"][0].as_str().unwrap())
}
}
}
" "
a."u-url"."u-uid"[href=post["properties"]["uid"][0].as_str().unwrap()] {
@if let Some(published) = post["properties"]["published"][0].as_str() {
time."dt-published"[datetime=published] {
@chrono::DateTime::parse_from_rfc3339(published)
.map(|dt| dt.format("on %a %b %e %T %Y").to_string())
.unwrap_or("sometime in the past".to_string())
}
} else {
"sometime in the past"
}
}
}
}
}
FullEntry<'a>(post: &'a serde_json::Value, from_feed: bool) {
article."h-entry" {
header.metadata {
@if let Some(name) = post["properties"]["name"][0].as_str() {
h1."p-name" { @name }
}
@if let Some(author) = post["properties"]["author"][0].as_object() {
section."mini-h-card" {
a.larger."u-author"[href=author["properties"]["uid"][0].as_str().unwrap()] {
@if let Some(photo) = author["properties"]["photo"][0].as_str() {
img[src=photo, loading="lazy"];
} else if let Some(photo) = author["properties"]["photo"][0].as_object() {
img[
src=photo["value"].as_str().unwrap(),
alt=photo["alt"].as_str().unwrap(),
loading="lazy"
];
}
@author["properties"]["name"][0].as_str().unwrap()
}
}
}
div {
span {
a."u-url"."u-uid"[href=post["properties"]["uid"][0].as_str().unwrap(), rel=(!from_feed).then_some("canonical")] {
@if let Some(published) = post["properties"]["published"][0].as_str() {
time."dt-published"[datetime=published] {
@chrono::DateTime::parse_from_rfc3339(published)
.map(|dt| dt.format("%a %b %e %T %Y").to_string())
.unwrap_or("sometime in the past".to_string())
}
}
}
}
@if post["properties"]["visibility"][0].as_str().unwrap_or("public") != "public" {
span."p-visibility"[value=post["properties"]["visibility"][0].as_str().unwrap()] {
@post["properties"]["visibility"][0].as_str().unwrap()
}
}
@if post["properties"]["category"].is_array() {
span {
ul.categories {
"Tagged: "
@for cat in post["properties"]["category"].as_array().unwrap() {
li."p-category" { @cat.as_str().unwrap() }
}
}
}
}
@if post["properties"]["in-reply-to"].is_array() {
span {
"In reply to: "
ul.replyctx {
@for ctx in post["properties"]["in-reply-to"].as_array().unwrap() {
@if let Some(ctx) = ctx.as_str() {
li {
a."u-in-reply-to"[href=ctx] {
@ctx.truncate_ellipse(48).as_ref()
}
}
} else if let Some(ctx) = ctx.as_object() {
li {
a."u-in-reply-to"[href=ctx["properties"]["uid"][0]
.as_str()
.unwrap_or_else(|| ctx["properties"]["url"][0].as_str().unwrap())]
{
@ctx["properties"]["uid"][0]
.as_str()
.unwrap_or_else(|| ctx["properties"]["url"][0].as_str().unwrap())
.truncate_ellipse(48)
.as_ref()
}
}
}
}
}
}
}
}
@if post["properties"]["url"].as_array().unwrap().len() > 1 {
hr;
ul {
"Pretty permalinks for this post:"
@for url in post["properties"]["url"].as_array().unwrap().iter().filter(|i| **i != post["properties"]["uid"][0]).map(|i| i.as_str().unwrap()) {
li {
a."u-url"[href=url] { @url }
}
}
}
}
@if let Some(links) = post["properties"]["syndication"].as_array() {
@if !links.is_empty() {
hr;
ul {
"Also published on:"
@for url in links.iter().filter_map(|i| i.as_str()) {
li { a."u-syndication"[href=url] { @url } }
}
}
}
}
@if post["properties"]["location"].is_array() || post["properties"]["checkin"].is_array() {
div {
@if post["properties"]["checkin"].is_array() {
span {
"Check-in to: "
@if post["properties"]["checkin"][0].is_string() {
// It's a URL
a."u-checkin"[href=post["properties"]["checkin"][0].as_str().unwrap()] {
@post["properties"]["checkin"][0].as_str().unwrap().truncate_ellipse(24).as_ref()
}
} else {
a."u-checkin"[href=post["properties"]["checkin"][0]["properties"]["uid"][0].as_str().unwrap()] {
@post["properties"]["checkin"][0]["properties"]["name"][0].as_str().unwrap()
}
}
}
}
@if post["properties"]["location"].is_array() {
span {
"Location: "
@if post["properties"]["location"][0].is_string() {
// It's a geo: URL
// We need to decode it
a."u-location"[href=post["properties"]["location"][0].as_str().unwrap()] {
@decode_geo_uri(post["properties"]["location"][0].as_str().unwrap())
}
} else {
// It's an inner h-geo object
a."u-location"[href=post["properties"]["location"][0]["value"].as_str().map(|x| x.to_string()).unwrap_or(format!("geo:{},{}", post["properties"]["location"][0]["properties"]["latitude"][0].as_str().unwrap(), post["properties"]["location"][0]["properties"]["longitude"][0].as_str().unwrap()))] {
// I'm a lazy bitch
@decode_geo_uri(&post["properties"]["location"][0]["value"].as_str().map(|x| x.to_string()).unwrap_or(format!("geo:{},{}", post["properties"]["location"][0]["properties"]["latitude"][0].as_str().unwrap(), post["properties"]["location"][0]["properties"]["longitude"][0].as_str().unwrap())))
}
}
}
}
}
}
@if post["properties"]["ate"].is_array() || post["properties"]["drank"].is_array() {
div {
@if let Some(ate) = post["properties"]["ate"].as_array() {
span { ul {
"Ate:"
@for food in ate {
li {
@if let Some(food) = food.as_str() {
// If this is a string, it's a URL.
a."u-ate"[href=food] {
@food.truncate_ellipse(24).as_ref()
}
} else {
// This is a rich food object (mm, sounds tasty! I wanna eat something tasty)
a."u-ate"[href=food["properties"]["uid"][0].as_str().unwrap_or("#")] {
@food["properties"]["name"][0].as_str()
.unwrap_or(food["properties"]["uid"][0].as_str().unwrap_or("#").truncate_ellipse(24).as_ref())
}
}
}
}
} }
}
@if let Some(drank) = post["properties"]["drank"].as_array() {
span { ul {
"Drank:"
@for food in drank {
li {
@if let Some(food) = food.as_str() {
// If this is a string, it's a URL.
a."u-drank"[href=food] {
@food.truncate_ellipse(24).as_ref()
}
} else {
// This is a rich food object (mm, sounds tasty! I wanna eat something tasty)
a."u-drank"[href=food["properties"]["uid"][0].as_str().unwrap_or("#")] {
@food["properties"]["name"][0].as_str()
.unwrap_or(food["properties"]["uid"][0].as_str().unwrap_or("#").truncate_ellipse(24).as_ref())
}
}
}
}
} }
}
}
}
}
@PhotoGallery { photos: post["properties"]["photo"].as_array() }
@if *from_feed {
@if let Some(summary) = post["properties"]["summary"][0].as_str() {
p."p-summary" { @summary }
a[href=post["properties"]["uid"][0].as_str().unwrap()] { "Read more.." }
} else if let Some(content) = post["properties"]["content"][0]["html"].as_str() {
// TODO: ellipsize content by showing only the first paragraph
main."e-content" {
@markup::raw(content.trim())
}
}
} else if let Some(content) = post["properties"]["content"][0]["html"].as_str() {
main."e-content" {
@markup::raw(content.trim())
}
}
@WebInteractions { post, from_feed: *from_feed }
}
}
VCard<'a>(card: &'a serde_json::Value) {
article."h-card" {
@if card["properties"]["photo"][0].is_string() {
img."u-photo"[src=card["properties"]["photo"][0].as_str().unwrap()];
} else if card["properties"]["photo"][0].is_object() {
img."u-photo"[
src=card["properties"]["photo"][0]["value"].as_str().unwrap(),
alt=card["properties"]["photo"][0]["alt"].as_str().unwrap()
];
}
h1 {
a."u-url"."u-uid"."p-name"[href=card["properties"]["uid"][0].as_str().unwrap()] {
@card["properties"]["name"][0].as_str().unwrap()
}
}
@if card["properties"]["pronoun"].is_array() {
span {
"("
@for (i, pronoun) in card["properties"]["pronoun"].as_array().unwrap().iter().filter_map(|v| v.as_str()).enumerate() {
span."p-pronoun" {
@pronoun
}
// Insert commas between multiple sets of pronouns
@if i < (card["properties"]["pronoun"].as_array().unwrap().len() - 1) {", "}
}
")"
}
}
@if card["properties"]["note"].is_array() {
p."p-note" {
@card["properties"]["note"][0]["value"].as_str().unwrap_or_else(|| card["properties"]["note"][0].as_str().unwrap())
}
}
@if card["properties"]["url"].is_array() {
ul {
"Can be found elsewhere at:"
@for url in card["properties"]["url"]
.as_array()
.unwrap()
.iter()
.filter_map(|v| v.as_str())
.filter(|v| v != &card["properties"]["uid"][0].as_str().unwrap())
.filter(|v| !card["properties"]["author"][0].as_str().is_some_and(|a| v.starts_with(a)))
{
li { a."u-url"[href=url, rel="me"] { @url } }
}
}
}
}
}
Food<'a>(food: &'a serde_json::Value) {
article."h-food" {
header.metadata {
h1 {
a."p-name"."u-url"[href=food["properties"]["url"][0].as_str().unwrap()] {
@food["properties"]["name"][0].as_str().unwrap()
}
}
}
@PhotoGallery { photos: food["properties"]["photo"].as_array() }
}
}
Feed<'a>(feed: &'a serde_json::Value, cursor: Option<&'a str>) {
div."h-feed" {
div.metadata {
@if feed["properties"]["name"][0].is_string() {
h1."p-name".titanic {
a[href=feed["properties"]["uid"][0].as_str().unwrap(), rel="feed"] {
@feed["properties"]["name"][0].as_str().unwrap()
}
}
}
}
@if feed["children"].is_array() {
@for child in feed["children"].as_array().unwrap() {
@match child["type"][0].as_str().unwrap() {
"h-entry" => { @Entry { post: child, from_feed: true, } }
"h-feed" => { @Feed { feed: child, cursor: None } }
"h-food" => { @Food { food: child } }
//"h-event" => { }
"h-card" => { @VCard { card: child } }
something_else => {
p {
"There's supposed to be an "
@something_else
" object here. But Kittybox can't render it right now."
small { "Sorry! TToTT" }
}
}
}
}
}
@if let Some(cursor) = cursor {
a[rel="prev", href=format!("{}?after={}", feed["properties"]["uid"][0].as_str().unwrap(), cursor)] {
"Older posts"
}
} else {
p {
"Looks like you reached the end. Wanna jump back to the "
a[href=feed["properties"]["uid"][0].as_str().unwrap()] {
"beginning"
} "?"
}
}
}
}
//=======================================
// Components library
//=======================================
PhotoGallery<'a>(photos: Option<&'a Vec<serde_json::Value>>) {
@if let Some(photos) = photos {
@for photo in photos.iter() {
@if let Some(photo) = photo.as_str() {
img."u-photo"[src=photo, loading="lazy"];
} else if photo.is_object() {
@if let Some(thumbnail) = photo["thumbnail"].as_str() {
a."u-photo"[href=photo["value"].as_str().unwrap()] {
img[src=thumbnail,
loading="lazy",
alt=photo["alt"].as_str().unwrap_or("")
];
}
} else {
img."u-photo"[src=photo["value"].as_str().unwrap(),
loading="lazy",
alt=photo["alt"].as_str().unwrap_or("")
];
}
}
}
}
}
WebInteractions<'a>(post: &'a serde_json::Value, from_feed: bool) {
footer.webinteractions {
p[style="display: none", "aria-hidden"="false"] {
"Webmention counters:"
}
ul.counters {
li {
span."icon like-icon"["aria-label"="likes"] {
span."like-icon-label"["aria-hidden"="true"] {
"❤️"
}
}
span.counter { @post["properties"]["like"].as_array().map(|a| a.len()).unwrap_or(0) }
}
li {
span.icon["aria-label"="replies"] { "💬" }
span.counter { @post["properties"]["comment"].as_array().map(|a| a.len()).unwrap_or(0) }
}
li {
span.icon["aria-label"="reposts"] { "🔄" }
span.counter { @post["properties"]["repost"].as_array().map(|a| a.len()).unwrap_or(0) }
}
li {
span.icon["aria-label"="bookmarks"] { "🔖" }
span.counter { @post["properties"]["bookmark"].as_array().map(|a| a.len()).unwrap_or(0) }
}
}
/*@if *from_feed && (
post["properties"]["like"].as_array().map(|a| a.len()).unwrap_or(0)
+ post["properties"]["bookmark"].as_array().map(|a| a.len()).unwrap_or(0)
+ post["properties"]["repost"].as_array().map(|a| a.len()).unwrap_or(0)
+ post["properties"]["comment"].as_array().map(|a| a.len()).unwrap_or(0)
) > 0 {
details {
summary { "Show comments and reactions" }
// TODO actually render facepiles and comments
@if let Some(likes) = post["properties"]["like"].as_array() {
@if !likes.is_empty() {
// Show a facepile of likes for a post
}
}
@if let Some(bookmarks) = post["properties"]["bookmark"].as_array() {
@if !bookmarks.is_empty() {
// Show a facepile of bookmarks for a post
}
}
@if let Some(reposts) = post["properties"]["repost"].as_array() {
@if !reposts.is_empty() {
// Show a facepile of reposts for a post
}
}
@if let Some(comments) = post["properties"]["comment"].as_array() {
@for comment in comments.iter() {
// Show all the comments recursively (so we could do Salmention with them)
}
}
}
}*/
}
}
}