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.
180 lines
5.4 KiB
180 lines
5.4 KiB
<script lang="ts"> |
|
import type { NostrEvent } from '../../types/nostr.js'; |
|
import { KIND } from '../../types/kind-lookup.js'; |
|
|
|
interface Props { |
|
event: NostrEvent; |
|
hideTitle?: boolean; // If true, don't show title (already displayed elsewhere) |
|
hideSummary?: boolean; // If true, don't show summary (already displayed elsewhere) |
|
} |
|
|
|
let { event, hideTitle = false, hideSummary = false }: Props = $props(); |
|
|
|
// Extract metadata tags (using $derived for reactivity) |
|
// Only show fields that are NOT in the event header (CardHeader) |
|
// Header shows: pubkey (author), title, summary, time, client, topics, view/reply buttons, menu |
|
// Images are shown via MediaAttachments |
|
// So MetadataCard should only show: description, author tag (if different from pubkey), and other tags |
|
const description = $derived(event.tags.find(t => t[0] === 'description' && t[1])?.[1]); |
|
const rawSummary = $derived(event.tags.find(t => t[0] === 'summary' && t[1])?.[1]); |
|
const summary = $derived(hideSummary ? null : rawSummary); // Hide if already in header |
|
const authorTag = $derived(event.tags.find(t => t[0] === 'author' && t[1])?.[1]); |
|
// Only show author tag if it's different from the event pubkey (which is already shown in header) |
|
const author = $derived(authorTag && authorTag !== event.pubkey ? authorTag : null); |
|
const title = $derived( |
|
hideTitle ? null : |
|
(event.tags.find(t => t[0] === 'title' && t[1])?.[1] || |
|
(() => { |
|
// Fallback to d-tag in Title Case |
|
const dTag = event.tags.find(t => t[0] === 'd' && t[1])?.[1]; |
|
if (dTag) { |
|
return dTag.split('-').map(word => |
|
word.charAt(0).toUpperCase() + word.slice(1).toLowerCase() |
|
).join(' '); |
|
} |
|
return null; |
|
})()) |
|
); |
|
|
|
const hasMetadata = $derived(description || summary || author || title); |
|
const hasContent = $derived(event.content && event.content.trim().length > 0); |
|
const shouldShowMetadata = $derived(hasMetadata || !hasContent); // Show metadata if it exists OR if there's no content |
|
|
|
// Media kinds check (for filtering tags display) |
|
const MEDIA_KINDS: number[] = [KIND.PICTURE_NOTE, KIND.VIDEO_NOTE, KIND.SHORT_VIDEO_NOTE, KIND.VOICE_NOTE, KIND.VOICE_REPLY]; |
|
const isMediaKind = $derived(MEDIA_KINDS.includes(event.kind)); |
|
</script> |
|
|
|
{#if shouldShowMetadata} |
|
<div class="metadata-card"> |
|
{#if title} |
|
<h2 class="metadata-title">{title}</h2> |
|
{/if} |
|
|
|
<div class="metadata-content"> |
|
{#if description} |
|
<p class="metadata-description">{description}</p> |
|
{/if} |
|
|
|
{#if summary} |
|
<p class="metadata-summary">{summary}</p> |
|
{/if} |
|
|
|
{#if author} |
|
<p class="metadata-author">Author: {author}</p> |
|
{/if} |
|
|
|
{#if !hasContent && !isMediaKind} |
|
<!-- Show all tags when there's no content, but skip for media kinds (MediaAttachments handles those) --> |
|
<div class="metadata-tags"> |
|
{#each event.tags as tag} |
|
{#if tag[0] !== 'image' && tag[0] !== 'description' && tag[0] !== 'summary' && tag[0] !== 'author' && tag[0] !== 'title' && tag[0] !== 'd' && tag[0] !== 'imeta' && tag[0] !== 'file' && tag[0] !== 'alt' && tag[0] !== 'x' && tag[0] !== 'm'} |
|
<div class="metadata-tag"> |
|
<span class="metadata-tag-name">{tag[0]}:</span> |
|
{#each tag.slice(1) as value} |
|
{#if value} |
|
<span class="metadata-tag-value">{value}</span> |
|
{/if} |
|
{/each} |
|
</div> |
|
{/if} |
|
{/each} |
|
</div> |
|
{/if} |
|
</div> |
|
</div> |
|
{/if} |
|
|
|
<style> |
|
.metadata-card { |
|
border: 1px solid var(--fog-border, #e5e7eb); |
|
border-radius: 0.5rem; |
|
padding: 1.5rem; |
|
margin-bottom: 1.5rem; |
|
background: var(--fog-post, #ffffff); |
|
} |
|
|
|
:global(.dark) .metadata-card { |
|
border-color: var(--fog-dark-border, #374151); |
|
background: var(--fog-dark-post, #1f2937); |
|
} |
|
|
|
.metadata-title { |
|
margin: 0 0 1rem 0; |
|
font-size: 1.5rem; |
|
font-weight: 600; |
|
color: var(--fog-text, #1f2937); |
|
} |
|
|
|
:global(.dark) .metadata-title { |
|
color: var(--fog-dark-text, #f9fafb); |
|
} |
|
|
|
.metadata-content { |
|
display: flex; |
|
flex-direction: column; |
|
gap: 0.75rem; |
|
} |
|
|
|
.metadata-description, |
|
.metadata-summary { |
|
margin: 0; |
|
color: var(--fog-text, #1f2937); |
|
line-height: 1.6; |
|
} |
|
|
|
:global(.dark) .metadata-description, |
|
:global(.dark) .metadata-summary { |
|
color: var(--fog-dark-text, #f9fafb); |
|
} |
|
|
|
.metadata-author { |
|
margin: 0; |
|
font-size: 0.875rem; |
|
color: var(--fog-text-light, #52667a); |
|
} |
|
|
|
:global(.dark) .metadata-author { |
|
color: var(--fog-dark-text-light, #a8b8d0); |
|
} |
|
|
|
.metadata-tags { |
|
margin-top: 1rem; |
|
padding-top: 1rem; |
|
border-top: 1px solid var(--fog-border, #e5e7eb); |
|
display: flex; |
|
flex-direction: column; |
|
gap: 0.5rem; |
|
} |
|
|
|
:global(.dark) .metadata-tags { |
|
border-top-color: var(--fog-dark-border, #374151); |
|
} |
|
|
|
.metadata-tag { |
|
display: flex; |
|
flex-wrap: wrap; |
|
gap: 0.5rem; |
|
align-items: baseline; |
|
font-size: 0.875rem; |
|
} |
|
|
|
.metadata-tag-name { |
|
font-weight: 600; |
|
color: var(--fog-text, #1f2937); |
|
} |
|
|
|
:global(.dark) .metadata-tag-name { |
|
color: var(--fog-dark-text, #f9fafb); |
|
} |
|
|
|
.metadata-tag-value { |
|
color: var(--fog-text-light, #52667a); |
|
font-family: monospace; |
|
word-break: break-all; |
|
} |
|
|
|
:global(.dark) .metadata-tag-value { |
|
color: var(--fog-dark-text-light, #a8b8d0); |
|
} |
|
</style>
|
|
|