upub/web/src/components/object.rs

81 lines
2.4 KiB
Rust
Raw Normal View History

use leptos::*;
use crate::{prelude::*, URL_SENSITIVE};
use apb::{target::Addressed, Base, Object};
#[component]
pub fn Object(object: serde_json::Value) -> impl IntoView {
let uid = object.id().unwrap_or_default().to_string();
let content = dissolve::strip_html_tags(object.content().unwrap_or_default());
let author_id = object.attributed_to().id().unwrap_or_default();
let author = CACHE.get_or(&author_id, serde_json::Value::String(author_id.clone()));
let sensitive = object.sensitive().unwrap_or_default();
let attachments = object.attachment()
.map(|x| {
let (expand, set_expand) = create_signal(false);
let href = x.url().id().unwrap_or_default();
view! {
<p class="center">
<img
class="attachment ml-1"
class:expand=expand
src={move || if sensitive && !expand.get() {
URL_SENSITIVE.to_string()
} else {
href.clone()
}}
title={x.name().unwrap_or_default().to_string()}
on:click=move |_| set_expand.set(!expand.get())
/>
</p>
}
})
.collect_view();
2024-04-23 23:28:19 +02:00
let attachments_padding = if object.attachment().is_empty() {
None
} else {
Some(view! { <div class="pb-1"></div> })
};
view! {
<table class="align w-100">
<tr>
<td><ActorBanner object=author /></td>
<td class="rev" >
{object.in_reply_to().id().map(|reply| view! {
<small><i><a class="clean" href={Uri::web(FetchKind::Object, &reply)} title={reply}>reply</a></i></small>
})}
<PrivacyMarker addressed=object.addressed() />
<a class="clean hover ml-s" href={Uri::web(FetchKind::Object, object.id().unwrap_or_default())}>
<DateTime t=object.published() />
</a>
<sup><small><a class="clean ml-s" href={uid} target="_blank">""</a></small></sup>
</td>
</tr>
</table>
<blockquote class="tl">
<Summary summary=object.summary().map(|x| x.to_string()) open=!sensitive >
2024-04-23 23:28:19 +02:00
{content.into_iter().map(|x| view! { <p>{x}</p> }).collect_view()}
{attachments_padding}
{attachments}
</Summary>
</blockquote>
2024-04-23 23:28:19 +02:00
}
}
#[component]
pub fn Summary(summary: Option<String>, open: bool, children: Children) -> impl IntoView {
2024-04-23 23:28:19 +02:00
match summary.filter(|x| !x.is_empty()) {
None => children().into_view(),
Some(summary) => view! {
<details class="pa-s" prop:open=open>
2024-04-23 23:28:19 +02:00
<summary>
<code class="cw color ml-s w-100">{summary}</code>
</summary>
{children()}
</details>
}.into_view(),
}
}