You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
164 lines
5.9 KiB
164 lines
5.9 KiB
import { ExtendedKind } from '@/constants' |
|
import { |
|
getGitRepublicRepoContext, |
|
gitRepublicRepoWebUrl, |
|
type GitRepublicRepoContext |
|
} from '@/lib/git-republic-event' |
|
import { cn } from '@/lib/utils' |
|
import { Event, nip19 } from 'nostr-tools' |
|
import { useMemo } from 'react' |
|
import { useTranslation } from 'react-i18next' |
|
import { ExternalLink, GitBranch, CircleDot, Tag } from 'lucide-react' |
|
import MarkdownArticle from './MarkdownArticle/MarkdownArticle' |
|
|
|
function repoHeadline(ctx: GitRepublicRepoContext): string { |
|
const name = ctx.displayName || ctx.repoId |
|
try { |
|
const npub = nip19.npubEncode(ctx.ownerHex) |
|
const short = `${npub.slice(0, 14)}…` |
|
return `${short} / ${name}` |
|
} catch { |
|
return name |
|
} |
|
} |
|
|
|
export default function GitRepublicEventCard({ |
|
event, |
|
className, |
|
variant = 'full' |
|
}: { |
|
event: Event |
|
className?: string |
|
variant?: 'full' | 'compact' |
|
}) { |
|
const { t } = useTranslation() |
|
const ctx = useMemo(() => getGitRepublicRepoContext(event), [event]) |
|
const webUrl = useMemo(() => (ctx ? gitRepublicRepoWebUrl(ctx) : null), [ctx]) |
|
|
|
const subject = event.tags.find((t) => t[0] === 'subject')?.[1] |
|
const titleTag = event.tags.find((t) => t[0] === 'title')?.[1] |
|
const tagName = event.tags.find((t) => t[0] === 'tag')?.[1] |
|
const description = |
|
event.kind === ExtendedKind.GIT_REPO_ANNOUNCEMENT |
|
? event.tags.find((t) => t[0] === 'description')?.[1] |
|
: undefined |
|
|
|
const isDraft = event.tags.some((t) => t[0] === 'draft' && t[1] === 'true') |
|
const isPrerelease = event.tags.some((t) => t[0] === 'prerelease' && t[1] === 'true') |
|
|
|
const { Icon, badge, headline } = useMemo(() => { |
|
if (event.kind === ExtendedKind.GIT_REPO_ANNOUNCEMENT) { |
|
const name = event.tags.find((t) => t[0] === 'name')?.[1] || ctx?.repoId || t('Git Republic repository') |
|
return { |
|
Icon: GitBranch, |
|
badge: t('Git Republic repository'), |
|
headline: name |
|
} |
|
} |
|
if (event.kind === ExtendedKind.GIT_ISSUE) { |
|
return { |
|
Icon: CircleDot, |
|
badge: t('Git Republic issue'), |
|
headline: subject || t('Git Republic issue') |
|
} |
|
} |
|
if (event.kind === ExtendedKind.GIT_RELEASE) { |
|
const h = titleTag || tagName || t('Git Republic release') |
|
return { Icon: Tag, badge: t('Git Republic release'), headline: h } |
|
} |
|
return { Icon: GitBranch, badge: t('Git Republic'), headline: t('Git Republic event') } |
|
}, [event, ctx?.repoId, subject, tagName, titleTag, t]) |
|
|
|
const body = |
|
event.kind === ExtendedKind.GIT_REPO_ANNOUNCEMENT |
|
? description || event.content |
|
: event.content |
|
|
|
const compact = variant === 'compact' |
|
|
|
return ( |
|
<div |
|
className={cn( |
|
'rounded-xl border border-violet-500/25 bg-gradient-to-br from-violet-500/[0.08] via-background to-sky-500/[0.06] shadow-sm', |
|
compact ? 'p-3' : 'p-4', |
|
className |
|
)} |
|
> |
|
<div className="flex items-start gap-3"> |
|
<div |
|
className={cn( |
|
'flex shrink-0 items-center justify-center rounded-lg bg-primary/10 text-primary', |
|
compact ? 'size-9' : 'size-10' |
|
)} |
|
aria-hidden |
|
> |
|
<Icon className={compact ? 'size-4' : 'size-5'} /> |
|
</div> |
|
<div className="min-w-0 flex-1 space-y-1"> |
|
<div className={cn('flex flex-wrap items-center gap-2', compact && 'sr-only')}> |
|
<span className="text-[0.65rem] font-semibold uppercase tracking-wider text-muted-foreground"> |
|
{badge} |
|
</span> |
|
{isDraft ? ( |
|
<span className="rounded-md bg-amber-500/15 px-1.5 py-0.5 text-[0.65rem] font-medium text-amber-700 dark:text-amber-400"> |
|
{t('Draft')} |
|
</span> |
|
) : null} |
|
{isPrerelease ? ( |
|
<span className="rounded-md bg-sky-500/15 px-1.5 py-0.5 text-[0.65rem] font-medium text-sky-700 dark:text-sky-400"> |
|
{t('Pre-release')} |
|
</span> |
|
) : null} |
|
</div> |
|
{compact && (isDraft || isPrerelease) ? ( |
|
<div className="flex flex-wrap gap-1"> |
|
{isDraft ? ( |
|
<span className="rounded bg-amber-500/15 px-1 py-0.5 text-[0.6rem] font-medium text-amber-700 dark:text-amber-400"> |
|
{t('Draft')} |
|
</span> |
|
) : null} |
|
{isPrerelease ? ( |
|
<span className="rounded bg-sky-500/15 px-1 py-0.5 text-[0.6rem] font-medium text-sky-700 dark:text-sky-400"> |
|
{t('Pre-release')} |
|
</span> |
|
) : null} |
|
</div> |
|
) : null} |
|
<h3 |
|
className={cn( |
|
'font-semibold leading-snug text-foreground break-words', |
|
compact ? 'text-sm' : 'text-base' |
|
)} |
|
> |
|
{headline} |
|
</h3> |
|
{event.kind === ExtendedKind.GIT_RELEASE && tagName ? ( |
|
<p className="font-mono text-xs text-muted-foreground">{tagName}</p> |
|
) : null} |
|
{ctx ? ( |
|
<p className="truncate text-xs text-muted-foreground" title={repoHeadline(ctx)}> |
|
{repoHeadline(ctx)} |
|
</p> |
|
) : null} |
|
{webUrl ? ( |
|
<a |
|
href={webUrl} |
|
target="_blank" |
|
rel="noreferrer noopener" |
|
className="inline-flex max-w-full items-center gap-1 text-xs font-medium text-primary hover:text-foreground hover:underline underline-offset-2 transition-colors" |
|
onClick={(e) => e.stopPropagation()} |
|
> |
|
<ExternalLink className="size-3 shrink-0" aria-hidden /> |
|
<span className="truncate">{t('Open in Git Republic')}</span> |
|
</a> |
|
) : null} |
|
</div> |
|
</div> |
|
{body.trim() ? ( |
|
<div className={cn(compact ? 'mt-2 line-clamp-4' : 'mt-3', 'min-w-0 text-sm')}> |
|
<MarkdownArticle event={{ ...event, content: body }} hideMetadata className="prose-sm" /> |
|
</div> |
|
) : null} |
|
</div> |
|
) |
|
}
|
|
|