Convert account, orders, and federation sexp_components.py to pure sexp() calls
Eliminates all f-string HTML from the remaining three services, completing the migration of all sexp_components.py files to the s-expression rendering system. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -23,10 +23,10 @@ def _social_nav_html(actor: Any) -> str:
|
||||
|
||||
if not actor:
|
||||
choose_url = url_for("identity.choose_username_form")
|
||||
return (
|
||||
'<nav class="flex gap-3 text-sm items-center">'
|
||||
f'<a href="{choose_url}" class="px-2 py-1 rounded hover:bg-stone-200 font-bold">Choose username</a>'
|
||||
'</nav>'
|
||||
return sexp(
|
||||
'(nav :class "flex gap-3 text-sm items-center"'
|
||||
' (a :href url :class "px-2 py-1 rounded hover:bg-stone-200 font-bold" "Choose username"))',
|
||||
url=choose_url,
|
||||
)
|
||||
|
||||
links = [
|
||||
@@ -38,27 +38,40 @@ def _social_nav_html(actor: Any) -> str:
|
||||
("social.search", "Search"),
|
||||
]
|
||||
|
||||
parts = ['<nav class="flex gap-3 text-sm items-center flex-wrap">']
|
||||
parts = []
|
||||
for endpoint, label in links:
|
||||
href = url_for(endpoint)
|
||||
bold = " font-bold" if request.path == href else ""
|
||||
parts.append(f'<a href="{href}" class="px-2 py-1 rounded hover:bg-stone-200{bold}">{label}</a>')
|
||||
parts.append(sexp(
|
||||
'(a :href href :class cls (raw! label))',
|
||||
href=href,
|
||||
cls=f"px-2 py-1 rounded hover:bg-stone-200{bold}",
|
||||
label=label,
|
||||
))
|
||||
|
||||
# Notifications with live badge
|
||||
notif_url = url_for("social.notifications")
|
||||
notif_count_url = url_for("social.notification_count")
|
||||
notif_bold = " font-bold" if request.path == notif_url else ""
|
||||
parts.append(
|
||||
f'<a href="{notif_url}" class="px-2 py-1 rounded hover:bg-stone-200 relative{notif_bold}">Notifications'
|
||||
f'<span hx-get="{notif_count_url}" hx-trigger="load, every 30s" hx-swap="innerHTML"'
|
||||
f' class="absolute -top-2 -right-3 text-xs bg-red-500 text-white rounded-full px-1 empty:hidden"></span></a>'
|
||||
)
|
||||
parts.append(sexp(
|
||||
'(a :href href :class cls "Notifications"'
|
||||
' (span :hx-get count-url :hx-trigger "load, every 30s" :hx-swap "innerHTML"'
|
||||
' :class "absolute -top-2 -right-3 text-xs bg-red-500 text-white rounded-full px-1 empty:hidden"))',
|
||||
href=notif_url, cls=f"px-2 py-1 rounded hover:bg-stone-200 relative{notif_bold}",
|
||||
**{"count-url": notif_count_url},
|
||||
))
|
||||
|
||||
# Profile link
|
||||
profile_url = url_for("activitypub.actor_profile", username=actor.preferred_username)
|
||||
parts.append(f'<a href="{profile_url}" class="px-2 py-1 rounded hover:bg-stone-200">@{actor.preferred_username}</a>')
|
||||
parts.append('</nav>')
|
||||
return "".join(parts)
|
||||
parts.append(sexp(
|
||||
'(a :href href :class "px-2 py-1 rounded hover:bg-stone-200" (raw! label))',
|
||||
href=profile_url, label=f"@{actor.preferred_username}",
|
||||
))
|
||||
|
||||
return sexp(
|
||||
'(nav :class "flex gap-3 text-sm items-center flex-wrap" (raw! items))',
|
||||
items="".join(parts),
|
||||
)
|
||||
|
||||
|
||||
def _social_header_html(actor: Any) -> str:
|
||||
@@ -80,7 +93,7 @@ def _social_page(ctx: dict, actor: Any, *, content_html: str,
|
||||
sh=_social_header_html(actor),
|
||||
)
|
||||
return full_page(ctx, header_rows_html=hdr, content_html=content_html,
|
||||
meta_html=meta_html or f'<title>{escape(title)}</title>')
|
||||
meta_html=meta_html or sexp('(title (raw! t))', t=escape(title)))
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
@@ -106,11 +119,11 @@ def _interaction_buttons_html(item: Any, actor: Any) -> str:
|
||||
if liked:
|
||||
like_action = url_for("social.unlike")
|
||||
like_cls = "text-red-500 hover:text-red-600"
|
||||
like_icon = "♥"
|
||||
like_icon = "\u2665"
|
||||
else:
|
||||
like_action = url_for("social.like")
|
||||
like_cls = "hover:text-red-500"
|
||||
like_icon = "♡"
|
||||
like_icon = "\u2661"
|
||||
|
||||
if boosted:
|
||||
boost_action = url_for("social.unboost")
|
||||
@@ -120,21 +133,37 @@ def _interaction_buttons_html(item: Any, actor: Any) -> str:
|
||||
boost_cls = "hover:text-green-600"
|
||||
|
||||
reply_url = url_for("social.compose_form", reply_to=oid) if oid else ""
|
||||
reply_html = f'<a href="{reply_url}" class="hover:text-stone-700">Reply</a>' if reply_url else ""
|
||||
reply_html = sexp(
|
||||
'(a :href url :class "hover:text-stone-700" "Reply")',
|
||||
url=reply_url,
|
||||
) if reply_url else ""
|
||||
|
||||
return (
|
||||
f'<div class="flex items-center gap-4 mt-3 text-sm text-stone-500">'
|
||||
f'<form hx-post="{like_action}" hx-target="{target}" hx-swap="innerHTML">'
|
||||
f'<input type="hidden" name="object_id" value="{oid}">'
|
||||
f'<input type="hidden" name="author_inbox" value="{ainbox}">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<button type="submit" class="flex items-center gap-1 {like_cls}"><span>{like_icon}</span> {lcount}</button></form>'
|
||||
f'<form hx-post="{boost_action}" hx-target="{target}" hx-swap="innerHTML">'
|
||||
f'<input type="hidden" name="object_id" value="{oid}">'
|
||||
f'<input type="hidden" name="author_inbox" value="{ainbox}">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<button type="submit" class="flex items-center gap-1 {boost_cls}"><span>↻</span> {bcount}</button></form>'
|
||||
f'{reply_html}</div>'
|
||||
like_form = sexp(
|
||||
'(form :hx-post action :hx-target target :hx-swap "innerHTML"'
|
||||
' (input :type "hidden" :name "object_id" :value oid)'
|
||||
' (input :type "hidden" :name "author_inbox" :value ainbox)'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (button :type "submit" :class cls (span (raw! icon)) " " (raw! count)))',
|
||||
action=like_action, target=target, oid=oid, ainbox=ainbox,
|
||||
csrf=csrf, cls=f"flex items-center gap-1 {like_cls}",
|
||||
icon=like_icon, count=str(lcount),
|
||||
)
|
||||
|
||||
boost_form = sexp(
|
||||
'(form :hx-post action :hx-target target :hx-swap "innerHTML"'
|
||||
' (input :type "hidden" :name "object_id" :value oid)'
|
||||
' (input :type "hidden" :name "author_inbox" :value ainbox)'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (button :type "submit" :class cls (span "\u21bb") " " (raw! count)))',
|
||||
action=boost_action, target=target, oid=oid, ainbox=ainbox,
|
||||
csrf=csrf, cls=f"flex items-center gap-1 {boost_cls}",
|
||||
count=str(bcount),
|
||||
)
|
||||
|
||||
return sexp(
|
||||
'(div :class "flex items-center gap-4 mt-3 text-sm text-stone-500"'
|
||||
' (raw! like) (raw! boost) (raw! reply))',
|
||||
like=like_form, boost=boost_form, reply=reply_html,
|
||||
)
|
||||
|
||||
|
||||
@@ -151,45 +180,74 @@ def _post_card_html(item: Any, actor: Any) -> str:
|
||||
url = getattr(item, "url", None)
|
||||
post_type = getattr(item, "post_type", "")
|
||||
|
||||
boost_html = f'<div class="text-sm text-stone-500 mb-2">Boosted by {escape(boosted_by)}</div>' if boosted_by else ""
|
||||
boost_html = sexp(
|
||||
'(div :class "text-sm text-stone-500 mb-2" "Boosted by " (raw! name))',
|
||||
name=str(escape(boosted_by)),
|
||||
) if boosted_by else ""
|
||||
|
||||
if actor_icon:
|
||||
avatar = f'<img src="{actor_icon}" alt="" class="w-10 h-10 rounded-full">'
|
||||
avatar = sexp(
|
||||
'(img :src src :alt "" :class "w-10 h-10 rounded-full")',
|
||||
src=actor_icon,
|
||||
)
|
||||
else:
|
||||
initial = actor_name[0].upper() if actor_name else "?"
|
||||
avatar = f'<div class="w-10 h-10 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold text-sm">{initial}</div>'
|
||||
avatar = sexp(
|
||||
'(div :class "w-10 h-10 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold text-sm" (raw! i))',
|
||||
i=initial,
|
||||
)
|
||||
|
||||
domain_html = f"@{escape(actor_domain)}" if actor_domain else ""
|
||||
time_html = published.strftime("%b %d, %H:%M") if published else ""
|
||||
|
||||
if summary:
|
||||
content_html = (
|
||||
f'<details class="mt-2"><summary class="text-stone-500 cursor-pointer">CW: {escape(summary)}</summary>'
|
||||
f'<div class="mt-2 prose prose-sm prose-stone max-w-none">{content}</div></details>'
|
||||
content_html = sexp(
|
||||
'(details :class "mt-2"'
|
||||
' (summary :class "text-stone-500 cursor-pointer" "CW: " (raw! s))'
|
||||
' (div :class "mt-2 prose prose-sm prose-stone max-w-none" (raw! c)))',
|
||||
s=str(escape(summary)), c=content,
|
||||
)
|
||||
else:
|
||||
content_html = f'<div class="mt-2 prose prose-sm prose-stone max-w-none">{content}</div>'
|
||||
content_html = sexp(
|
||||
'(div :class "mt-2 prose prose-sm prose-stone max-w-none" (raw! c))',
|
||||
c=content,
|
||||
)
|
||||
|
||||
original_html = ""
|
||||
if url and post_type == "remote":
|
||||
original_html = f'<a href="{url}" target="_blank" rel="noopener" class="text-sm text-stone-400 hover:underline mt-1 inline-block">original</a>'
|
||||
original_html = sexp(
|
||||
'(a :href url :target "_blank" :rel "noopener"'
|
||||
' :class "text-sm text-stone-400 hover:underline mt-1 inline-block" "original")',
|
||||
url=url,
|
||||
)
|
||||
|
||||
interactions_html = ""
|
||||
if actor:
|
||||
oid = getattr(item, "object_id", "") or ""
|
||||
safe_id = oid.replace("/", "_").replace(":", "_")
|
||||
interactions_html = f'<div id="interactions-{safe_id}">{_interaction_buttons_html(item, actor)}</div>'
|
||||
interactions_html = sexp(
|
||||
'(div :id id (raw! buttons))',
|
||||
id=f"interactions-{safe_id}",
|
||||
buttons=_interaction_buttons_html(item, actor),
|
||||
)
|
||||
|
||||
return (
|
||||
f'<article class="bg-white rounded-lg shadow-sm border border-stone-200 p-4 mb-4">'
|
||||
f'{boost_html}'
|
||||
f'<div class="flex items-start gap-3">{avatar}'
|
||||
f'<div class="flex-1 min-w-0">'
|
||||
f'<div class="flex items-baseline gap-2">'
|
||||
f'<span class="font-semibold text-stone-900">{escape(actor_name)}</span>'
|
||||
f'<span class="text-sm text-stone-500">@{escape(actor_username)}{domain_html}</span>'
|
||||
f'<span class="text-sm text-stone-400 ml-auto">{time_html}</span></div>'
|
||||
f'{content_html}{original_html}{interactions_html}</div></div></article>'
|
||||
return sexp(
|
||||
'(article :class "bg-white rounded-lg shadow-sm border border-stone-200 p-4 mb-4"'
|
||||
' (raw! boost)'
|
||||
' (div :class "flex items-start gap-3"'
|
||||
' (raw! avatar)'
|
||||
' (div :class "flex-1 min-w-0"'
|
||||
' (div :class "flex items-baseline gap-2"'
|
||||
' (span :class "font-semibold text-stone-900" (raw! aname))'
|
||||
' (span :class "text-sm text-stone-500" "@" (raw! ausername) (raw! domain))'
|
||||
' (span :class "text-sm text-stone-400 ml-auto" (raw! time)))'
|
||||
' (raw! content) (raw! original) (raw! interactions))))',
|
||||
boost=boost_html, avatar=avatar,
|
||||
aname=str(escape(actor_name)),
|
||||
ausername=str(escape(actor_username)),
|
||||
domain=domain_html, time=time_html,
|
||||
content=content_html, original=original_html,
|
||||
interactions=interactions_html,
|
||||
)
|
||||
|
||||
|
||||
@@ -211,7 +269,10 @@ def _timeline_items_html(items: list, timeline_type: str, actor: Any,
|
||||
next_url = url_for("social.actor_timeline_page", id=actor_id, before=before)
|
||||
else:
|
||||
next_url = url_for(f"social.{timeline_type}_timeline_page", before=before)
|
||||
parts.append(f'<div hx-get="{next_url}" hx-trigger="revealed" hx-swap="outerHTML"></div>')
|
||||
parts.append(sexp(
|
||||
'(div :hx-get url :hx-trigger "revealed" :hx-swap "outerHTML")',
|
||||
url=next_url,
|
||||
))
|
||||
|
||||
return "".join(parts)
|
||||
|
||||
@@ -238,48 +299,75 @@ def _actor_card_html(a: Any, actor: Any, followed_urls: set,
|
||||
safe_id = actor_url.replace("/", "_").replace(":", "_")
|
||||
|
||||
if icon_url:
|
||||
avatar = f'<img src="{icon_url}" alt="" class="w-12 h-12 rounded-full">'
|
||||
avatar = sexp(
|
||||
'(img :src src :alt "" :class "w-12 h-12 rounded-full")',
|
||||
src=icon_url,
|
||||
)
|
||||
else:
|
||||
initial = (display_name or username)[0].upper() if (display_name or username) else "?"
|
||||
avatar = f'<div class="w-12 h-12 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold">{initial}</div>'
|
||||
avatar = sexp(
|
||||
'(div :class "w-12 h-12 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold" (raw! i))',
|
||||
i=initial,
|
||||
)
|
||||
|
||||
# Name link
|
||||
if list_type == "following" and aid:
|
||||
name_html = f'<a href="{url_for("social.actor_timeline", id=aid)}" class="font-semibold text-stone-900 hover:underline">{escape(display_name)}</a>'
|
||||
elif list_type == "search" and aid:
|
||||
name_html = f'<a href="{url_for("social.actor_timeline", id=aid)}" class="font-semibold text-stone-900 hover:underline">{escape(display_name)}</a>'
|
||||
if (list_type in ("following", "search")) and aid:
|
||||
name_html = sexp(
|
||||
'(a :href href :class "font-semibold text-stone-900 hover:underline" (raw! name))',
|
||||
href=url_for("social.actor_timeline", id=aid),
|
||||
name=str(escape(display_name)),
|
||||
)
|
||||
else:
|
||||
name_html = f'<a href="https://{domain}/@{username}" target="_blank" rel="noopener" class="font-semibold text-stone-900 hover:underline">{escape(display_name)}</a>'
|
||||
name_html = sexp(
|
||||
'(a :href href :target "_blank" :rel "noopener"'
|
||||
' :class "font-semibold text-stone-900 hover:underline" (raw! name))',
|
||||
href=f"https://{domain}/@{username}",
|
||||
name=str(escape(display_name)),
|
||||
)
|
||||
|
||||
summary_html = f'<div class="text-sm text-stone-600 mt-1 truncate">{summary}</div>' if summary else ""
|
||||
summary_html = sexp(
|
||||
'(div :class "text-sm text-stone-600 mt-1 truncate" (raw! s))',
|
||||
s=summary,
|
||||
) if summary else ""
|
||||
|
||||
# Follow/unfollow button
|
||||
button_html = ""
|
||||
if actor:
|
||||
is_followed = actor_url in (followed_urls or set())
|
||||
if list_type == "following" or is_followed:
|
||||
button_html = (
|
||||
f'<div class="flex-shrink-0"><form method="post" action="{url_for("social.unfollow")}"'
|
||||
f' hx-post="{url_for("social.unfollow")}" hx-target="closest article" hx-swap="outerHTML">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<input type="hidden" name="actor_url" value="{actor_url}">'
|
||||
f'<button type="submit" class="text-sm border border-stone-300 rounded px-3 py-1 hover:bg-stone-100">Unfollow</button></form></div>'
|
||||
button_html = sexp(
|
||||
'(div :class "flex-shrink-0"'
|
||||
' (form :method "post" :action action :hx-post action :hx-target "closest article" :hx-swap "outerHTML"'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (input :type "hidden" :name "actor_url" :value aurl)'
|
||||
' (button :type "submit" :class "text-sm border border-stone-300 rounded px-3 py-1 hover:bg-stone-100" "Unfollow")))',
|
||||
action=url_for("social.unfollow"), csrf=csrf, aurl=actor_url,
|
||||
)
|
||||
else:
|
||||
label = "Follow Back" if list_type == "followers" else "Follow"
|
||||
button_html = (
|
||||
f'<div class="flex-shrink-0"><form method="post" action="{url_for("social.follow")}"'
|
||||
f' hx-post="{url_for("social.follow")}" hx-target="closest article" hx-swap="outerHTML">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<input type="hidden" name="actor_url" value="{actor_url}">'
|
||||
f'<button type="submit" class="text-sm bg-stone-800 text-white rounded px-3 py-1 hover:bg-stone-700">{label}</button></form></div>'
|
||||
button_html = sexp(
|
||||
'(div :class "flex-shrink-0"'
|
||||
' (form :method "post" :action action :hx-post action :hx-target "closest article" :hx-swap "outerHTML"'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (input :type "hidden" :name "actor_url" :value aurl)'
|
||||
' (button :type "submit" :class "text-sm bg-stone-800 text-white rounded px-3 py-1 hover:bg-stone-700" (raw! label))))',
|
||||
action=url_for("social.follow"), csrf=csrf, aurl=actor_url, label=label,
|
||||
)
|
||||
|
||||
return (
|
||||
f'<article class="bg-white rounded-lg shadow-sm border border-stone-200 p-4 mb-3 flex items-center gap-4" id="actor-{safe_id}">'
|
||||
f'{avatar}<div class="flex-1 min-w-0">{name_html}'
|
||||
f'<div class="text-sm text-stone-500">@{escape(username)}@{escape(domain)}</div>'
|
||||
f'{summary_html}</div>{button_html}</article>'
|
||||
return sexp(
|
||||
'(article :class cls :id id'
|
||||
' (raw! avatar)'
|
||||
' (div :class "flex-1 min-w-0"'
|
||||
' (raw! name-link)'
|
||||
' (div :class "text-sm text-stone-500" "@" (raw! username) "@" (raw! domain))'
|
||||
' (raw! summary))'
|
||||
' (raw! button))',
|
||||
cls="bg-white rounded-lg shadow-sm border border-stone-200 p-4 mb-3 flex items-center gap-4",
|
||||
id=f"actor-{safe_id}",
|
||||
avatar=avatar,
|
||||
**{"name-link": name_html},
|
||||
username=str(escape(username)), domain=str(escape(domain)),
|
||||
summary=summary_html, button=button_html,
|
||||
)
|
||||
|
||||
|
||||
@@ -291,7 +379,10 @@ def _search_results_html(actors: list, query: str, page: int,
|
||||
parts = [_actor_card_html(a, actor, followed_urls, list_type="search") for a in actors]
|
||||
if len(actors) >= 20:
|
||||
next_url = url_for("social.search_page", q=query, page=page + 1)
|
||||
parts.append(f'<div hx-get="{next_url}" hx-trigger="revealed" hx-swap="outerHTML"></div>')
|
||||
parts.append(sexp(
|
||||
'(div :hx-get url :hx-trigger "revealed" :hx-swap "outerHTML")',
|
||||
url=next_url,
|
||||
))
|
||||
return "".join(parts)
|
||||
|
||||
|
||||
@@ -303,7 +394,10 @@ def _actor_list_items_html(actors: list, page: int, list_type: str,
|
||||
parts = [_actor_card_html(a, actor, followed_urls, list_type=list_type) for a in actors]
|
||||
if len(actors) >= 20:
|
||||
next_url = url_for(f"social.{list_type}_list_page", page=page + 1)
|
||||
parts.append(f'<div hx-get="{next_url}" hx-trigger="revealed" hx-swap="outerHTML"></div>')
|
||||
parts.append(sexp(
|
||||
'(div :hx-get url :hx-trigger "revealed" :hx-swap "outerHTML")',
|
||||
url=next_url,
|
||||
))
|
||||
return "".join(parts)
|
||||
|
||||
|
||||
@@ -326,10 +420,16 @@ def _notification_html(notif: Any) -> str:
|
||||
border = " border-l-4 border-l-stone-400" if not read else ""
|
||||
|
||||
if from_icon:
|
||||
avatar = f'<img src="{from_icon}" alt="" class="w-8 h-8 rounded-full">'
|
||||
avatar = sexp(
|
||||
'(img :src src :alt "" :class "w-8 h-8 rounded-full")',
|
||||
src=from_icon,
|
||||
)
|
||||
else:
|
||||
initial = from_name[0].upper() if from_name else "?"
|
||||
avatar = f'<div class="w-8 h-8 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold text-xs">{initial}</div>'
|
||||
avatar = sexp(
|
||||
'(div :class "w-8 h-8 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold text-xs" (raw! i))',
|
||||
i=initial,
|
||||
)
|
||||
|
||||
domain_html = f"@{escape(from_domain)}" if from_domain else ""
|
||||
|
||||
@@ -344,16 +444,29 @@ def _notification_html(notif: Any) -> str:
|
||||
if ntype == "follow" and app_domain and app_domain != "federation":
|
||||
action += f" on {escape(app_domain)}"
|
||||
|
||||
preview_html = f'<div class="text-sm text-stone-500 mt-1 truncate">{escape(preview)}</div>' if preview else ""
|
||||
preview_html = sexp(
|
||||
'(div :class "text-sm text-stone-500 mt-1 truncate" (raw! p))',
|
||||
p=str(escape(preview)),
|
||||
) if preview else ""
|
||||
time_html = created.strftime("%b %d, %H:%M") if created else ""
|
||||
|
||||
return (
|
||||
f'<div class="bg-white rounded-lg shadow-sm border border-stone-200 p-4{border}">'
|
||||
f'<div class="flex items-start gap-3">{avatar}<div class="flex-1">'
|
||||
f'<div class="text-sm"><span class="font-semibold">{escape(from_name)}</span>'
|
||||
f' <span class="text-stone-500">@{escape(from_username)}{domain_html}</span>'
|
||||
f' <span class="text-stone-600">{action}</span></div>'
|
||||
f'{preview_html}<div class="text-xs text-stone-400 mt-1">{time_html}</div></div></div></div>'
|
||||
return sexp(
|
||||
'(div :class cls'
|
||||
' (div :class "flex items-start gap-3"'
|
||||
' (raw! avatar)'
|
||||
' (div :class "flex-1"'
|
||||
' (div :class "text-sm"'
|
||||
' (span :class "font-semibold" (raw! fname))'
|
||||
' " " (span :class "text-stone-500" "@" (raw! fusername) (raw! fdomain))'
|
||||
' " " (span :class "text-stone-600" (raw! action)))'
|
||||
' (raw! preview)'
|
||||
' (div :class "text-xs text-stone-400 mt-1" (raw! time)))))',
|
||||
cls=f"bg-white rounded-lg shadow-sm border border-stone-200 p-4{border}",
|
||||
avatar=avatar,
|
||||
fname=str(escape(from_name)),
|
||||
fusername=str(escape(from_username)),
|
||||
fdomain=domain_html, action=action,
|
||||
preview=preview_html, time=time_html,
|
||||
)
|
||||
|
||||
|
||||
@@ -381,21 +494,31 @@ async def render_login_page(ctx: dict) -> str:
|
||||
action = url_for("auth.start_login")
|
||||
csrf = generate_csrf_token()
|
||||
|
||||
error_html = f'<div class="bg-red-50 border border-red-200 text-red-700 p-3 rounded mb-4">{error}</div>' if error else ""
|
||||
content = (
|
||||
f'<div class="py-8 max-w-md mx-auto"><h1 class="text-2xl font-bold mb-6">Sign in</h1>{error_html}'
|
||||
f'<form method="post" action="{action}" class="space-y-4">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<div><label for="email" class="block text-sm font-medium mb-1">Email address</label>'
|
||||
f'<input type="email" name="email" id="email" value="{escape(email)}" required autofocus'
|
||||
f' class="w-full border border-stone-300 rounded px-3 py-2 focus:outline-none focus:ring-2 focus:ring-stone-500"></div>'
|
||||
f'<button type="submit" class="w-full bg-stone-800 text-white py-2 px-4 rounded hover:bg-stone-700 transition">'
|
||||
f'Send magic link</button></form></div>'
|
||||
error_html = sexp(
|
||||
'(div :class "bg-red-50 border border-red-200 text-red-700 p-3 rounded mb-4" (raw! e))',
|
||||
e=error,
|
||||
) if error else ""
|
||||
|
||||
content = sexp(
|
||||
'(div :class "py-8 max-w-md mx-auto"'
|
||||
' (h1 :class "text-2xl font-bold mb-6" "Sign in")'
|
||||
' (raw! err)'
|
||||
' (form :method "post" :action action :class "space-y-4"'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (div'
|
||||
' (label :for "email" :class "block text-sm font-medium mb-1" "Email address")'
|
||||
' (input :type "email" :name "email" :id "email" :value email :required true :autofocus true'
|
||||
' :class "w-full border border-stone-300 rounded px-3 py-2 focus:outline-none focus:ring-2 focus:ring-stone-500"))'
|
||||
' (button :type "submit"'
|
||||
' :class "w-full bg-stone-800 text-white py-2 px-4 rounded hover:bg-stone-700 transition"'
|
||||
' "Send magic link")))',
|
||||
err=error_html, action=action, csrf=csrf,
|
||||
email=str(escape(email)),
|
||||
)
|
||||
|
||||
hdr = root_header_html(ctx)
|
||||
return full_page(ctx, header_rows_html=hdr, content_html=content,
|
||||
meta_html="<title>Login \u2014 Rose Ash</title>")
|
||||
meta_html='<title>Login \u2014 Rose Ash</title>')
|
||||
|
||||
|
||||
async def render_check_email_page(ctx: dict) -> str:
|
||||
@@ -403,18 +526,18 @@ async def render_check_email_page(ctx: dict) -> str:
|
||||
email = ctx.get("email", "")
|
||||
email_error = ctx.get("email_error")
|
||||
|
||||
error_html = ""
|
||||
if email_error:
|
||||
error_html = (
|
||||
f'<div class="bg-yellow-50 border border-yellow-200 text-yellow-700 p-3 rounded mt-4">'
|
||||
f'{escape(email_error)}</div>'
|
||||
)
|
||||
content = (
|
||||
'<div class="py-8 max-w-md mx-auto text-center">'
|
||||
'<h1 class="text-2xl font-bold mb-4">Check your email</h1>'
|
||||
f'<p class="text-stone-600 mb-2">We sent a sign-in link to <strong>{escape(email)}</strong>.</p>'
|
||||
'<p class="text-stone-500 text-sm">Click the link in the email to sign in. The link expires in 15 minutes.</p>'
|
||||
f'{error_html}</div>'
|
||||
error_html = sexp(
|
||||
'(div :class "bg-yellow-50 border border-yellow-200 text-yellow-700 p-3 rounded mt-4" (raw! e))',
|
||||
e=str(escape(email_error)),
|
||||
) if email_error else ""
|
||||
|
||||
content = sexp(
|
||||
'(div :class "py-8 max-w-md mx-auto text-center"'
|
||||
' (h1 :class "text-2xl font-bold mb-4" "Check your email")'
|
||||
' (p :class "text-stone-600 mb-2" "We sent a sign-in link to " (strong (raw! email)) ".")'
|
||||
' (p :class "text-stone-500 text-sm" "Click the link in the email to sign in. The link expires in 15 minutes.")'
|
||||
' (raw! err))',
|
||||
email=str(escape(email)), err=error_html,
|
||||
)
|
||||
|
||||
hdr = root_header_html(ctx)
|
||||
@@ -435,14 +558,19 @@ async def render_timeline_page(ctx: dict, items: list, timeline_type: str,
|
||||
compose_html = ""
|
||||
if actor:
|
||||
compose_url = url_for("social.compose_form")
|
||||
compose_html = f'<a href="{compose_url}" class="bg-stone-800 text-white px-4 py-2 rounded hover:bg-stone-700">Compose</a>'
|
||||
compose_html = sexp(
|
||||
'(a :href url :class "bg-stone-800 text-white px-4 py-2 rounded hover:bg-stone-700" "Compose")',
|
||||
url=compose_url,
|
||||
)
|
||||
|
||||
timeline_html = _timeline_items_html(items, timeline_type, actor)
|
||||
|
||||
content = (
|
||||
f'<div class="flex items-center justify-between mb-6">'
|
||||
f'<h1 class="text-2xl font-bold">{label} Timeline</h1>{compose_html}</div>'
|
||||
f'<div id="timeline">{timeline_html}</div>'
|
||||
content = sexp(
|
||||
'(div :class "flex items-center justify-between mb-6"'
|
||||
' (h1 :class "text-2xl font-bold" (raw! label) " Timeline")'
|
||||
' (raw! compose))'
|
||||
'(div :id "timeline" (raw! tl))',
|
||||
label=label, compose=compose_html, tl=timeline_html,
|
||||
)
|
||||
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
@@ -469,23 +597,27 @@ async def render_compose_page(ctx: dict, actor: Any, reply_to: str | None) -> st
|
||||
|
||||
reply_html = ""
|
||||
if reply_to:
|
||||
reply_html = (
|
||||
f'<input type="hidden" name="in_reply_to" value="{escape(reply_to)}">'
|
||||
f'<div class="text-sm text-stone-500">Replying to <span class="font-mono">{escape(reply_to)}</span></div>'
|
||||
reply_html = sexp(
|
||||
'(input :type "hidden" :name "in_reply_to" :value val)'
|
||||
'(div :class "text-sm text-stone-500" "Replying to " (span :class "font-mono" (raw! rt)))',
|
||||
val=str(escape(reply_to)), rt=str(escape(reply_to)),
|
||||
)
|
||||
|
||||
content = (
|
||||
f'<h1 class="text-2xl font-bold mb-6">Compose</h1>'
|
||||
f'<form method="post" action="{action}" class="space-y-4">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">{reply_html}'
|
||||
f'<textarea name="content" rows="6" maxlength="5000" required'
|
||||
f' class="w-full border border-stone-300 rounded-lg p-3 focus:outline-none focus:ring-2 focus:ring-stone-500"'
|
||||
f' placeholder="What\'s on your mind?"></textarea>'
|
||||
f'<div class="flex items-center justify-between">'
|
||||
f'<select name="visibility" class="border border-stone-300 rounded px-3 py-1.5 text-sm">'
|
||||
f'<option value="public">Public</option><option value="unlisted">Unlisted</option>'
|
||||
f'<option value="followers">Followers only</option></select>'
|
||||
f'<button type="submit" class="bg-stone-800 text-white px-6 py-2 rounded hover:bg-stone-700">Publish</button></div></form>'
|
||||
content = sexp(
|
||||
'(h1 :class "text-2xl font-bold mb-6" "Compose")'
|
||||
'(form :method "post" :action action :class "space-y-4"'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (raw! reply)'
|
||||
' (textarea :name "content" :rows "6" :maxlength "5000" :required true'
|
||||
' :class "w-full border border-stone-300 rounded-lg p-3 focus:outline-none focus:ring-2 focus:ring-stone-500"'
|
||||
' :placeholder "What\'s on your mind?")'
|
||||
' (div :class "flex items-center justify-between"'
|
||||
' (select :name "visibility" :class "border border-stone-300 rounded px-3 py-1.5 text-sm"'
|
||||
' (option :value "public" "Public")'
|
||||
' (option :value "unlisted" "Unlisted")'
|
||||
' (option :value "followers" "Followers only"))'
|
||||
' (button :type "submit" :class "bg-stone-800 text-white px-6 py-2 rounded hover:bg-stone-700" "Publish")))',
|
||||
action=action, csrf=csrf, reply=reply_html,
|
||||
)
|
||||
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
@@ -509,19 +641,30 @@ async def render_search_page(ctx: dict, query: str, actors: list, total: int,
|
||||
info_html = ""
|
||||
if query and total:
|
||||
s = "s" if total != 1 else ""
|
||||
info_html = f'<p class="text-sm text-stone-500 mb-4">{total} result{s} for <strong>{escape(query)}</strong></p>'
|
||||
info_html = sexp(
|
||||
'(p :class "text-sm text-stone-500 mb-4" (raw! t))',
|
||||
t=f"{total} result{s} for <strong>{escape(query)}</strong>",
|
||||
)
|
||||
elif query:
|
||||
info_html = f'<p class="text-stone-500 mb-4">No results found for <strong>{escape(query)}</strong></p>'
|
||||
info_html = sexp(
|
||||
'(p :class "text-stone-500 mb-4" (raw! t))',
|
||||
t=f"No results found for <strong>{escape(query)}</strong>",
|
||||
)
|
||||
|
||||
content = (
|
||||
f'<h1 class="text-2xl font-bold mb-6">Search</h1>'
|
||||
f'<form method="get" action="{search_url}" class="mb-6"'
|
||||
f' hx-get="{search_page_url}" hx-target="#search-results" hx-push-url="{search_url}">'
|
||||
f'<div class="flex gap-2"><input type="text" name="q" value="{escape(query)}"'
|
||||
f' class="flex-1 border border-stone-300 rounded-lg px-4 py-2 focus:outline-none focus:ring-2 focus:ring-stone-500"'
|
||||
f' placeholder="Search users or @user@instance.tld">'
|
||||
f'<button type="submit" class="bg-stone-800 text-white px-6 py-2 rounded hover:bg-stone-700">Search</button></div></form>'
|
||||
f'{info_html}<div id="search-results">{results_html}</div>'
|
||||
content = sexp(
|
||||
'(h1 :class "text-2xl font-bold mb-6" "Search")'
|
||||
'(form :method "get" :action search-url :class "mb-6"'
|
||||
' :hx-get search-page-url :hx-target "#search-results" :hx-push-url search-url'
|
||||
' (div :class "flex gap-2"'
|
||||
' (input :type "text" :name "q" :value query'
|
||||
' :class "flex-1 border border-stone-300 rounded-lg px-4 py-2 focus:outline-none focus:ring-2 focus:ring-stone-500"'
|
||||
' :placeholder "Search users or @user@instance.tld")'
|
||||
' (button :type "submit" :class "bg-stone-800 text-white px-6 py-2 rounded hover:bg-stone-700" "Search")))'
|
||||
'(raw! info)'
|
||||
'(div :id "search-results" (raw! results))',
|
||||
**{"search-url": search_url, "search-page-url": search_page_url},
|
||||
query=str(escape(query)),
|
||||
info=info_html, results=results_html,
|
||||
)
|
||||
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
@@ -542,9 +685,11 @@ async def render_following_page(ctx: dict, actors: list, total: int,
|
||||
actor: Any) -> str:
|
||||
"""Full page: following list."""
|
||||
items_html = _actor_list_items_html(actors, 1, "following", set(), actor)
|
||||
content = (
|
||||
f'<h1 class="text-2xl font-bold mb-6">Following <span class="text-stone-400 font-normal">({total})</span></h1>'
|
||||
f'<div id="actor-list">{items_html}</div>'
|
||||
content = sexp(
|
||||
'(h1 :class "text-2xl font-bold mb-6" "Following "'
|
||||
' (span :class "text-stone-400 font-normal" "(" (raw! total) ")"))'
|
||||
'(div :id "actor-list" (raw! items))',
|
||||
total=str(total), items=items_html,
|
||||
)
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
title="Following \u2014 Rose Ash")
|
||||
@@ -559,9 +704,11 @@ async def render_followers_page(ctx: dict, actors: list, total: int,
|
||||
followed_urls: set, actor: Any) -> str:
|
||||
"""Full page: followers list."""
|
||||
items_html = _actor_list_items_html(actors, 1, "followers", followed_urls, actor)
|
||||
content = (
|
||||
f'<h1 class="text-2xl font-bold mb-6">Followers <span class="text-stone-400 font-normal">({total})</span></h1>'
|
||||
f'<div id="actor-list">{items_html}</div>'
|
||||
content = sexp(
|
||||
'(h1 :class "text-2xl font-bold mb-6" "Followers "'
|
||||
' (span :class "text-stone-400 font-normal" "(" (raw! total) ")"))'
|
||||
'(div :id "actor-list" (raw! items))',
|
||||
total=str(total), items=items_html,
|
||||
)
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
title="Followers \u2014 Rose Ash")
|
||||
@@ -590,39 +737,61 @@ async def render_actor_timeline_page(ctx: dict, remote_actor: Any, items: list,
|
||||
actor_url = getattr(remote_actor, "actor_url", "")
|
||||
|
||||
if icon_url:
|
||||
avatar = f'<img src="{icon_url}" alt="" class="w-16 h-16 rounded-full">'
|
||||
avatar = sexp(
|
||||
'(img :src src :alt "" :class "w-16 h-16 rounded-full")',
|
||||
src=icon_url,
|
||||
)
|
||||
else:
|
||||
initial = display_name[0].upper() if display_name else "?"
|
||||
avatar = f'<div class="w-16 h-16 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold text-xl">{initial}</div>'
|
||||
avatar = sexp(
|
||||
'(div :class "w-16 h-16 rounded-full bg-stone-300 flex items-center justify-center text-stone-600 font-bold text-xl" (raw! i))',
|
||||
i=initial,
|
||||
)
|
||||
|
||||
summary_html = f'<div class="text-sm text-stone-600 mt-2">{summary}</div>' if summary else ""
|
||||
summary_html = sexp(
|
||||
'(div :class "text-sm text-stone-600 mt-2" (raw! s))',
|
||||
s=summary,
|
||||
) if summary else ""
|
||||
|
||||
follow_html = ""
|
||||
if actor:
|
||||
if is_following:
|
||||
follow_html = (
|
||||
f'<div class="flex-shrink-0"><form method="post" action="{url_for("social.unfollow")}">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<input type="hidden" name="actor_url" value="{actor_url}">'
|
||||
f'<button type="submit" class="border border-stone-300 rounded px-4 py-2 hover:bg-stone-100">Unfollow</button></form></div>'
|
||||
follow_html = sexp(
|
||||
'(div :class "flex-shrink-0"'
|
||||
' (form :method "post" :action action'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (input :type "hidden" :name "actor_url" :value aurl)'
|
||||
' (button :type "submit" :class "border border-stone-300 rounded px-4 py-2 hover:bg-stone-100" "Unfollow")))',
|
||||
action=url_for("social.unfollow"), csrf=csrf, aurl=actor_url,
|
||||
)
|
||||
else:
|
||||
follow_html = (
|
||||
f'<div class="flex-shrink-0"><form method="post" action="{url_for("social.follow")}">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<input type="hidden" name="actor_url" value="{actor_url}">'
|
||||
f'<button type="submit" class="bg-stone-800 text-white rounded px-4 py-2 hover:bg-stone-700">Follow</button></form></div>'
|
||||
follow_html = sexp(
|
||||
'(div :class "flex-shrink-0"'
|
||||
' (form :method "post" :action action'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (input :type "hidden" :name "actor_url" :value aurl)'
|
||||
' (button :type "submit" :class "bg-stone-800 text-white rounded px-4 py-2 hover:bg-stone-700" "Follow")))',
|
||||
action=url_for("social.follow"), csrf=csrf, aurl=actor_url,
|
||||
)
|
||||
|
||||
timeline_html = _timeline_items_html(items, "actor", actor, remote_actor.id)
|
||||
|
||||
content = (
|
||||
f'<div class="bg-white rounded-lg shadow-sm border border-stone-200 p-6 mb-6">'
|
||||
f'<div class="flex items-center gap-4">{avatar}'
|
||||
f'<div class="flex-1"><h1 class="text-xl font-bold">{escape(display_name)}</h1>'
|
||||
f'<div class="text-stone-500">@{escape(remote_actor.preferred_username)}@{escape(remote_actor.domain)}</div>'
|
||||
f'{summary_html}</div>{follow_html}</div></div>'
|
||||
f'<div id="timeline">{timeline_html}</div>'
|
||||
content = sexp(
|
||||
'(div :class "bg-white rounded-lg shadow-sm border border-stone-200 p-6 mb-6"'
|
||||
' (div :class "flex items-center gap-4"'
|
||||
' (raw! avatar)'
|
||||
' (div :class "flex-1"'
|
||||
' (h1 :class "text-xl font-bold" (raw! dname))'
|
||||
' (div :class "text-stone-500" "@" (raw! username) "@" (raw! domain))'
|
||||
' (raw! summary))'
|
||||
' (raw! follow)))'
|
||||
'(div :id "timeline" (raw! tl))',
|
||||
avatar=avatar,
|
||||
dname=str(escape(display_name)),
|
||||
username=str(escape(remote_actor.preferred_username)),
|
||||
domain=str(escape(remote_actor.domain)),
|
||||
summary=summary_html, follow=follow_html,
|
||||
tl=timeline_html,
|
||||
)
|
||||
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
@@ -643,11 +812,17 @@ async def render_notifications_page(ctx: dict, notifications: list,
|
||||
actor: Any) -> str:
|
||||
"""Full page: notifications."""
|
||||
if not notifications:
|
||||
notif_html = '<p class="text-stone-500">No notifications yet.</p>'
|
||||
notif_html = sexp('(p :class "text-stone-500" "No notifications yet.")')
|
||||
else:
|
||||
notif_html = '<div class="space-y-2">' + "".join(_notification_html(n) for n in notifications) + '</div>'
|
||||
notif_html = sexp(
|
||||
'(div :class "space-y-2" (raw! items))',
|
||||
items="".join(_notification_html(n) for n in notifications),
|
||||
)
|
||||
|
||||
content = f'<h1 class="text-2xl font-bold mb-6">Notifications</h1>{notif_html}'
|
||||
content = sexp(
|
||||
'(h1 :class "text-2xl font-bold mb-6" "Notifications") (raw! notifs)',
|
||||
notifs=notif_html,
|
||||
)
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
title="Notifications \u2014 Rose Ash")
|
||||
|
||||
@@ -669,25 +844,37 @@ async def render_choose_username_page(ctx: dict) -> str:
|
||||
check_url = url_for("identity.check_username")
|
||||
actor = ctx.get("actor")
|
||||
|
||||
error_html = f'<div class="bg-red-50 border border-red-200 text-red-700 p-3 rounded mb-4">{error}</div>' if error else ""
|
||||
error_html = sexp(
|
||||
'(div :class "bg-red-50 border border-red-200 text-red-700 p-3 rounded mb-4" (raw! e))',
|
||||
e=error,
|
||||
) if error else ""
|
||||
|
||||
content = (
|
||||
f'<div class="py-8 max-w-md mx-auto">'
|
||||
f'<h1 class="text-2xl font-bold mb-2">Choose your username</h1>'
|
||||
f'<p class="text-stone-600 mb-6">This will be your identity on the fediverse: '
|
||||
f'<strong>@username@{escape(ap_domain)}</strong></p>'
|
||||
f'{error_html}'
|
||||
f'<form method="post" class="space-y-4">'
|
||||
f'<input type="hidden" name="csrf_token" value="{csrf}">'
|
||||
f'<div><label for="username" class="block text-sm font-medium mb-1">Username</label>'
|
||||
f'<div class="flex items-center"><span class="text-stone-400 mr-1">@</span>'
|
||||
f'<input type="text" name="username" id="username" value="{escape(username)}"'
|
||||
f' pattern="[a-z][a-z0-9_]{{2,31}}" minlength="3" maxlength="32" required autocomplete="off"'
|
||||
f' class="flex-1 border border-stone-300 rounded px-3 py-2 focus:outline-none focus:ring-2 focus:ring-stone-500"'
|
||||
f' hx-get="{check_url}" hx-trigger="keyup changed delay:300ms" hx-target="#username-status" hx-include="[name=\'username\']">'
|
||||
f'</div><div id="username-status" class="text-sm mt-1"></div>'
|
||||
f'<p class="text-xs text-stone-400 mt-1">3-32 characters. Lowercase letters, numbers, underscores. Must start with a letter.</p></div>'
|
||||
f'<button type="submit" class="w-full bg-stone-800 text-white py-2 px-4 rounded hover:bg-stone-700 transition">Claim username</button></form></div>'
|
||||
content = sexp(
|
||||
'(div :class "py-8 max-w-md mx-auto"'
|
||||
' (h1 :class "text-2xl font-bold mb-2" "Choose your username")'
|
||||
' (p :class "text-stone-600 mb-6" "This will be your identity on the fediverse: "'
|
||||
' (strong "@username@" (raw! domain)))'
|
||||
' (raw! err)'
|
||||
' (form :method "post" :class "space-y-4"'
|
||||
' (input :type "hidden" :name "csrf_token" :value csrf)'
|
||||
' (div'
|
||||
' (label :for "username" :class "block text-sm font-medium mb-1" "Username")'
|
||||
' (div :class "flex items-center"'
|
||||
' (span :class "text-stone-400 mr-1" "@")'
|
||||
' (input :type "text" :name "username" :id "username" :value uname'
|
||||
' :pattern "[a-z][a-z0-9_]{2,31}" :minlength "3" :maxlength "32"'
|
||||
' :required true :autocomplete "off"'
|
||||
' :class "flex-1 border border-stone-300 rounded px-3 py-2 focus:outline-none focus:ring-2 focus:ring-stone-500"'
|
||||
' :hx-get check-url :hx-trigger "keyup changed delay:300ms" :hx-target "#username-status"'
|
||||
' :hx-include "[name=\'username\']"))'
|
||||
' (div :id "username-status" :class "text-sm mt-1")'
|
||||
' (p :class "text-xs text-stone-400 mt-1" "3-32 characters. Lowercase letters, numbers, underscores. Must start with a letter."))'
|
||||
' (button :type "submit"'
|
||||
' :class "w-full bg-stone-800 text-white py-2 px-4 rounded hover:bg-stone-700 transition"'
|
||||
' "Claim username")))',
|
||||
domain=str(escape(ap_domain)), err=error_html,
|
||||
csrf=csrf, uname=str(escape(username)),
|
||||
**{"check-url": check_url},
|
||||
)
|
||||
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
@@ -705,29 +892,49 @@ async def render_profile_page(ctx: dict, actor: Any, activities: list,
|
||||
|
||||
ap_domain = config().get("ap_domain", "rose-ash.com")
|
||||
display_name = actor.display_name or actor.preferred_username
|
||||
summary_html = f'<p class="mt-2">{escape(actor.summary)}</p>' if actor.summary else ""
|
||||
summary_html = sexp(
|
||||
'(p :class "mt-2" (raw! s))',
|
||||
s=str(escape(actor.summary)),
|
||||
) if actor.summary else ""
|
||||
|
||||
activities_html = ""
|
||||
if activities:
|
||||
parts = []
|
||||
for a in activities:
|
||||
published = a.published.strftime("%Y-%m-%d %H:%M") if a.published else ""
|
||||
obj_type = f'<span class="text-sm text-stone-500">{a.object_type}</span>' if a.object_type else ""
|
||||
parts.append(
|
||||
f'<div class="bg-white rounded-lg shadow p-4"><div class="flex justify-between items-start">'
|
||||
f'<span class="font-medium">{a.activity_type}</span>'
|
||||
f'<span class="text-sm text-stone-400">{published}</span></div>{obj_type}</div>'
|
||||
)
|
||||
activities_html = '<div class="space-y-4">' + "".join(parts) + '</div>'
|
||||
obj_type_html = sexp(
|
||||
'(span :class "text-sm text-stone-500" (raw! t))',
|
||||
t=a.object_type,
|
||||
) if a.object_type else ""
|
||||
parts.append(sexp(
|
||||
'(div :class "bg-white rounded-lg shadow p-4"'
|
||||
' (div :class "flex justify-between items-start"'
|
||||
' (span :class "font-medium" (raw! atype))'
|
||||
' (span :class "text-sm text-stone-400" (raw! pub)))'
|
||||
' (raw! otype))',
|
||||
atype=a.activity_type, pub=published,
|
||||
otype=obj_type_html,
|
||||
))
|
||||
activities_html = sexp(
|
||||
'(div :class "space-y-4" (raw! items))',
|
||||
items="".join(parts),
|
||||
)
|
||||
else:
|
||||
activities_html = '<p class="text-stone-500">No activities yet.</p>'
|
||||
activities_html = sexp('(p :class "text-stone-500" "No activities yet.")')
|
||||
|
||||
content = (
|
||||
f'<div class="py-8"><div class="bg-white rounded-lg shadow p-6 mb-6">'
|
||||
f'<h1 class="text-2xl font-bold">{escape(display_name)}</h1>'
|
||||
f'<p class="text-stone-500">@{escape(actor.preferred_username)}@{escape(ap_domain)}</p>'
|
||||
f'{summary_html}</div>'
|
||||
f'<h2 class="text-xl font-bold mb-4">Activities ({total})</h2>{activities_html}</div>'
|
||||
content = sexp(
|
||||
'(div :class "py-8"'
|
||||
' (div :class "bg-white rounded-lg shadow p-6 mb-6"'
|
||||
' (h1 :class "text-2xl font-bold" (raw! dname))'
|
||||
' (p :class "text-stone-500" "@" (raw! username) "@" (raw! domain))'
|
||||
' (raw! summary))'
|
||||
' (h2 :class "text-xl font-bold mb-4" "Activities (" (raw! total) ")")'
|
||||
' (raw! activities))',
|
||||
dname=str(escape(display_name)),
|
||||
username=str(escape(actor.preferred_username)),
|
||||
domain=str(escape(ap_domain)),
|
||||
summary=summary_html,
|
||||
total=str(total), activities=activities_html,
|
||||
)
|
||||
|
||||
return _social_page(ctx, actor, content_html=content,
|
||||
|
||||
Reference in New Issue
Block a user