wildebeest/frontend/src/components/Status/index.tsx

83 wiersze
2.7 KiB
TypeScript

import { component$, $, useStyles$ } from '@builder.io/qwik'
import { Link, useNavigate } from '@builder.io/qwik-city'
import { formatTimeAgo } from '~/utils/dateTime'
import { Avatar } from '../avatar'
import type { Account, MastodonStatus } from '~/types'
import styles from '../../utils/innerHtmlContent.scss?inline'
import { MediaGallery } from '../MediaGallery.tsx'
type Props = {
status: MastodonStatus
}
export default component$((props: Props) => {
useStyles$(styles)
const nav = useNavigate()
const status = props.status.reblog ?? props.status
const reblogger = props.status.reblog && props.status.account
const accountUrl = `/@${status.account.username}`
const statusUrl = `${accountUrl}/${status.id}`
const handleContentClick = $(() => nav(statusUrl))
return (
<article class="p-4 border-t border-wildebeest-700 pointer">
<RebloggerLink account={reblogger}></RebloggerLink>
<div onClick$={handleContentClick}>
<div class="flex justify-between mb-3">
<div class="flex">
<Avatar primary={status.account} secondary={reblogger} />
<div class="flex-col ml-3">
<div>
<Link class="no-underline" href={accountUrl}>
{status.account.display_name}
</Link>
</div>
<div class="text-wildebeest-500">@{status.account.username}</div>
</div>
</div>
<Link class="no-underline" href={statusUrl}>
<div class="text-wildebeest-500 flex items-baseline">
<i style={{ height: '0.75rem', width: '0.75rem' }} class="fa fa-xs fa-globe w-3 h-3" />
<span class="ml-2 text-sm hover:underline">{formatTimeAgo(new Date(status.created_at))}</span>
</div>
</Link>
</div>
<div class="leading-relaxed inner-html-content" dangerouslySetInnerHTML={status.content} />
</div>
<MediaGallery medias={status.media_attachments} />
{status.card && status.media_attachments.length === 0 && (
<a class="no-underline" href={status.card.url}>
<div class="rounded flex border border-wildebeest-600">
<img class="w-16 h-16" src={status.card.image} />
<div class="p-3 overflow-hidden">
<div class="overflow-ellipsis text-sm text-bold text-wildebeest-400">{status.card.title}</div>
<div class="overflow-ellipsis mt-2 text-sm text-wildebeest-500">{status.card.provider_name}</div>
</div>
</div>
</a>
)}
</article>
)
})
export const RebloggerLink = ({ account }: { account: Account | null }) => {
return (
account && (
<div class="flex text-wildebeest-500 py-3">
<p>
<i class="fa fa-retweet mr-3 w-4 inline-block" />
<a class="no-underline" href={account.url}>
{account.display_name}
</a>
&nbsp;boosted
</p>
</div>
)
)
}