2024-04-17 22:07:47 +02:00
use apb ::{ Actor , Base , Collection , Object } ;
use leptos ::* ;
use leptos_router ::* ;
use crate ::prelude ::* ;
#[ component ]
pub fn AboutPage ( ) -> impl IntoView {
view! {
< div >
< Breadcrumb > about < / Breadcrumb >
< div class = " mt-s mb-s " >
< p > < code > μ pub < / code > " is a micro social network powered by " < a href = " " > ActivityPub < / a > < / p >
2024-04-17 23:07:56 +02:00
< p > < i > " the " < a href = " https://en.wikipedia.org/wiki/Fediverse " > fediverse < / a > " is an ensemble of social networks, which, while independently hosted, can communicate with each other " < / i > < / p >
< p > content is aggregated in timelines , logged out users can only access global server timeline < / p >
< / div >
< / div >
}
}
#[ component ]
pub fn ConfigPage ( ) -> impl IntoView {
view! {
< div >
< Breadcrumb > config < / Breadcrumb >
< div class = " mt-s mb-s " >
< p > < code > " not implemented :( " < / code > < / p >
2024-04-17 22:07:47 +02:00
< / div >
< / div >
}
}
#[ component ]
2024-04-19 06:58:54 +02:00
pub fn UserPage ( tl : Timeline ) -> impl IntoView {
2024-04-17 22:07:47 +02:00
let params = use_params_map ( ) ;
let auth = use_context ::< Auth > ( ) . expect ( " missing auth context " ) ;
2024-04-18 03:34:41 +02:00
let id = params . get ( ) . get ( " id " ) . cloned ( ) . unwrap_or_default ( ) ;
let _id = id . clone ( ) ; // wtf triple clone??? TODO!!
let actor = create_local_resource ( move | | _id . clone ( ) , move | id | {
2024-04-17 22:07:47 +02:00
async move {
2024-04-21 18:56:25 +02:00
match CACHE . get ( & Uri ::full ( FetchKind ::User , & id ) ) {
2024-04-17 22:07:47 +02:00
Some ( x ) = > Some ( x . clone ( ) ) ,
None = > {
2024-04-21 18:56:25 +02:00
let user : serde_json ::Value = Http ::fetch ( & Uri ::api ( FetchKind ::User , & id , true ) , auth ) . await . ok ( ) ? ;
CACHE . put ( Uri ::full ( FetchKind ::User , & id ) , user . clone ( ) ) ;
2024-04-17 22:07:47 +02:00
Some ( user )
} ,
}
}
} ) ;
view! {
< div >
< Breadcrumb back = true > users ::view < / Breadcrumb >
< div >
{ move | | match actor . get ( ) {
None = > view! { < p > loading .. . < / p > } . into_view ( ) ,
Some ( None ) = > view! { < p > < code > error loading < / code > < / p > } . into_view ( ) ,
2024-04-18 03:34:41 +02:00
Some ( Some ( object ) ) = > {
let uid = object . id ( ) . unwrap_or_default ( ) . to_string ( ) ;
let avatar_url = object . icon ( ) . get ( ) . map ( | x | x . url ( ) . id ( ) . unwrap_or_default ( ) ) . unwrap_or_default ( ) ;
let background_url = object . image ( ) . get ( ) . map ( | x | x . url ( ) . id ( ) . unwrap_or_default ( ) ) . unwrap_or_default ( ) ;
let display_name = object . name ( ) . unwrap_or_default ( ) . to_string ( ) ;
let username = object . preferred_username ( ) . unwrap_or_default ( ) . to_string ( ) ;
let summary = object . summary ( ) . unwrap_or_default ( ) . to_string ( ) ;
let domain = object . id ( ) . unwrap_or_default ( ) . replace ( " https:// " , " " ) . split ( '/' ) . next ( ) . unwrap_or_default ( ) . to_string ( ) ;
let actor_type = object . actor_type ( ) . unwrap_or ( apb ::ActorType ::Person ) ;
let actor_type_tag = if actor_type = = apb ::ActorType ::Person { None } else {
Some ( view! { < sup class = " ml-s " > < small > " [ " { actor_type . as_ref ( ) . to_lowercase ( ) } " ] " < / small > < / sup > } )
} ;
let created = object . published ( ) ;
2024-04-18 05:31:04 +02:00
let following = object . following ( ) . get ( ) . map ( | x | x . total_items ( ) . unwrap_or ( 0 ) ) . unwrap_or ( 0 ) ;
let followers = object . followers ( ) . get ( ) . map ( | x | x . total_items ( ) . unwrap_or ( 0 ) ) . unwrap_or ( 0 ) ;
let statuses = object . outbox ( ) . get ( ) . map ( | x | x . total_items ( ) . unwrap_or ( 0 ) ) . unwrap_or ( 0 ) ;
2024-04-21 18:56:25 +02:00
let tl_url = format! ( " {} /outbox/page " , Uri ::api ( FetchKind ::User , & id . clone ( ) , false ) ) ;
2024-04-19 06:58:54 +02:00
if ! tl . next . get ( ) . starts_with ( & tl_url ) {
tl . reset ( tl_url ) ;
}
2024-04-18 03:34:41 +02:00
view! {
< div class = " ml-3 mr-3 " >
< div
class = " banner "
style = { format! ( " background: center / cover url( {background_url} ); " ) }
>
// <table class="align w-100">
// <tr><td rowspan=3>
// <img src=
// </table>
< div style = " height: 10em " > < / div >
< / div >
< div class = " overlap " >
< table class = " pl-2 pr-2 align w-100 " style = " table-layout: fixed " >
< tr >
< td rowspan = 4 style = " width: 8em " >
< img class = " avatar-circle avatar-border mr-s " src = { avatar_url } style = " height: 7em; width: 7em " / >
< / td >
< td rowspan = 2 class = " bottom " >
< b class = " big " > { display_name } < / b > { actor_type_tag }
< / td >
2024-04-18 05:31:04 +02:00
< td rowspan = 2 class = " bottom rev " title = " statuses " > { statuses } " " < span class = " emoji " > " \u{1f582} " < / span > < / td >
2024-04-18 03:34:41 +02:00
< / tr >
< tr > < / tr >
< tr >
< td class = " top " >
< small > < a class = " clean hover " href = { uid } target = " _blank " > { username . clone ( ) } @ { domain } < / a > < / small >
< / td >
2024-04-18 05:31:04 +02:00
< td class = " rev " title = " following " > { following } " " < span class = " emoji " > " 👥 " < / span > < / td >
2024-04-18 03:34:41 +02:00
< / tr >
< tr >
< td >
< DateTime t = created / >
< / td >
2024-04-18 05:31:04 +02:00
< td class = " rev " title = " followers " > { followers } " " < span class = " emoji " > " 📢 " < / span > < / td >
2024-04-18 03:34:41 +02:00
< / tr >
< / table >
< blockquote class = " ml-2 mt-1 " > {
dissolve ::strip_html_tags ( & summary )
2024-04-17 22:07:47 +02:00
. into_iter ( )
2024-04-18 03:34:41 +02:00
. map ( | x | view! { < div > { x } < / div > } )
2024-04-17 22:07:47 +02:00
. collect_view ( )
2024-04-18 03:34:41 +02:00
} < / blockquote >
< / div >
< / div >
2024-04-19 06:58:54 +02:00
< TimelineFeed tl = tl / >
2024-04-18 03:34:41 +02:00
} . into_view ( )
} ,
2024-04-17 22:07:47 +02:00
} }
< / div >
< / div >
}
}
#[ component ]
2024-04-19 06:59:34 +02:00
pub fn ObjectPage ( tl : Timeline ) -> impl IntoView {
2024-04-17 22:07:47 +02:00
let params = use_params_map ( ) ;
let auth = use_context ::< Auth > ( ) . expect ( " missing auth context " ) ;
let object = create_local_resource ( move | | params . get ( ) . get ( " id " ) . cloned ( ) . unwrap_or_default ( ) , move | oid | {
async move {
2024-04-21 18:56:25 +02:00
match CACHE . get ( & Uri ::full ( FetchKind ::Object , & oid ) ) {
2024-04-17 22:07:47 +02:00
Some ( x ) = > Some ( x . clone ( ) ) ,
None = > {
2024-04-21 18:56:25 +02:00
let obj = Http ::fetch ::< serde_json ::Value > ( & Uri ::api ( FetchKind ::Object , & oid , true ) , auth ) . await . ok ( ) ? ;
CACHE . put ( Uri ::full ( FetchKind ::Object , & oid ) , obj . clone ( ) ) ;
2024-04-17 22:07:47 +02:00
Some ( obj )
}
}
}
} ) ;
view! {
< div >
< Breadcrumb back = true > objects ::view < / Breadcrumb >
< div class = " ma-2 " >
{ move | | match object . get ( ) {
None = > view! { < p > loading .. . < / p > } . into_view ( ) ,
2024-04-19 06:59:34 +02:00
Some ( None ) = > view! { < p > < code > loading failed < / code > < / p > } . into_view ( ) ,
Some ( Some ( o ) ) = > {
2024-04-21 18:56:25 +02:00
let object = o . clone ( ) ;
let tl_url = format! ( " {} /page " , Uri ::api ( FetchKind ::Context , & o . context ( ) . id ( ) . unwrap_or_default ( ) , false ) ) ;
2024-04-19 06:59:34 +02:00
if ! tl . next . get ( ) . starts_with ( & tl_url ) {
tl . reset ( tl_url ) ;
}
view! {
2024-04-21 18:56:25 +02:00
< Object object = object / >
2024-04-19 06:59:34 +02:00
< div class = " ml-1 mr-1 mt-2 " >
2024-04-22 01:01:20 +02:00
< TimelineReplies tl = tl root = o . id ( ) . unwrap_or_default ( ) . to_string ( ) / >
2024-04-19 06:59:34 +02:00
< / div >
} . into_view ( )
} ,
2024-04-17 22:07:47 +02:00
} }
< / div >
< / div >
}
}
#[ component ]
pub fn TimelinePage ( name : & 'static str , tl : Timeline ) -> impl IntoView {
let auth = use_context ::< Auth > ( ) . expect ( " missing auth context " ) ;
view! {
< div >
< Breadcrumb back = false >
{ name }
< a class = " clean ml-1 " href = " # " on :click = move | _ | {
tl . reset ( tl . next . get ( ) . split ( '?' ) . next ( ) . unwrap_or_default ( ) . to_string ( ) ) ;
spawn_local ( async move {
if let Err ( e ) = tl . more ( auth ) . await {
tracing ::error! ( " error fetching more items for timeline: {e} " ) ;
}
} )
} > < span class = " emoji " >
" \u{1f5d8} "
< / span > < / a >
< / Breadcrumb >
< div class = " mt-s mb-s " >
< TimelineFeed tl = tl / >
< / div >
< / div >
}
}
2024-04-21 19:37:20 +02:00
#[ component ]
pub fn DebugPage ( ) -> impl IntoView {
let ( object , set_object ) = create_signal ( serde_json ::Value ::String (
" use this view to fetch remote AP objects and inspect their content " . into ( ) )
) ;
2024-04-22 01:01:20 +02:00
let cached_ref : NodeRef < html ::Input > = create_node_ref ( ) ;
2024-04-21 19:37:20 +02:00
let auth = use_context ::< Auth > ( ) . expect ( " missing auth context " ) ;
2024-04-22 02:32:09 +02:00
let ( query , set_query ) = create_signal ( " " . to_string ( ) ) ;
2024-04-21 19:37:20 +02:00
view! {
< div >
< Breadcrumb back = true > debug < / Breadcrumb >
< div class = " mt-1 " >
2024-04-21 23:41:30 +02:00
< form on :submit = move | ev | {
ev . prevent_default ( ) ;
2024-04-22 01:01:20 +02:00
let cached = cached_ref . get ( ) . map ( | x | x . checked ( ) ) . unwrap_or_default ( ) ;
2024-04-22 02:32:09 +02:00
let fetch_url = query . get ( ) ;
2024-04-22 01:01:20 +02:00
if cached {
match CACHE . get ( & fetch_url ) {
Some ( x ) = > set_object . set ( x ) ,
None = > set_object . set ( serde_json ::Value ::String ( " not in cache! " . into ( ) ) ) ,
2024-04-21 23:41:30 +02:00
}
2024-04-22 01:01:20 +02:00
} else {
let url = format! ( " {URL_BASE} /dbg?id= {fetch_url} " ) ;
spawn_local ( async move {
match Http ::fetch ::< serde_json ::Value > ( & url , auth ) . await {
Ok ( x ) = > set_object . set ( x ) ,
Err ( e ) = > set_object . set ( serde_json ::Value ::String ( e . to_string ( ) ) ) ,
}
} ) ;
}
2024-04-21 23:41:30 +02:00
} >
2024-04-21 19:37:20 +02:00
< table class = " align w-100 " >
< tr >
2024-04-22 02:32:09 +02:00
< td class = " w-100 " > < input class = " w-100 " type = " text " on :input = move | ev | set_query . set ( event_target_value ( & ev ) ) placeholder = " AP id " / > < / td >
< td >
< small > < a
href = { move | | Uri ::web ( FetchKind ::Object , & query . get ( ) ) }
> obj < / a >
" "
< a
href = { move | | Uri ::web ( FetchKind ::User , & query . get ( ) ) }
> usr < / a > < / small >
< / td >
2024-04-21 23:41:30 +02:00
< td > < input type = " submit " class = " w-100 " value = " fetch " / > < / td >
2024-04-22 01:01:20 +02:00
< td > < input type = " checkbox " title = " cached " value = " cached " node_ref = cached_ref / > < / td >
2024-04-21 19:37:20 +02:00
< / tr >
< / table >
2024-04-21 23:41:30 +02:00
< / form >
2024-04-21 19:37:20 +02:00
< / div >
< pre class = " ma-1 " >
{ move | | serde_json ::to_string_pretty ( & object . get ( ) ) . unwrap_or ( " unserializable " . to_string ( ) ) }
< / pre >
< / div >
}
}