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.
 
 
 
 
 

100 lines
3.3 KiB

<script lang="ts">
import type { NostrEvent } from '../../types/nostr.js';
import { nostrClient } from '../../services/nostr/nostr-client.js';
import { relayManager } from '../../services/nostr/relay-manager.js';
import { stripMarkdown } from '../../services/text-utils.js';
interface Props {
parentEvent?: NostrEvent; // Optional - if not provided, will load by parentEventId
parentEventId?: string; // Optional - used to load parent if not provided
targetId?: string; // Optional ID to scroll to (defaults to parent event ID)
onParentLoaded?: (event: NostrEvent) => void; // Callback when parent is loaded
onOpenEvent?: (event: NostrEvent) => void; // Callback to open event in drawer
}
let { parentEvent: providedParentEvent, parentEventId, targetId, onParentLoaded, onOpenEvent }: Props = $props();
let loadedParentEvent = $state<NostrEvent | null>(null);
let loadingParent = $state(false);
// Derive the effective parent event: prefer provided, fall back to loaded
let parentEvent = $derived(providedParentEvent || loadedParentEvent);
// Sync provided parent event changes and load if needed
$effect(() => {
if (providedParentEvent) {
// If provided parent event is available, use it
return;
}
// If no provided parent event and we have an ID, try to load it
if (!loadedParentEvent && parentEventId && !loadingParent) {
loadParentEvent();
}
});
async function loadParentEvent() {
const eventId = parentEventId || parentEvent?.id;
if (!eventId || loadingParent) return;
loadingParent = true;
try {
const relays = relayManager.getFeedReadRelays();
const events = await nostrClient.fetchEvents(
[{ kinds: [KIND.SHORT_TEXT_NOTE], ids: [eventId] }],
relays,
{ useCache: true, cacheResults: true }
);
if (events.length > 0) {
loadedParentEvent = events[0];
if (onParentLoaded && typeof onParentLoaded === 'function') {
onParentLoaded(loadedParentEvent);
}
}
} catch (error) {
console.error('Error loading parent event:', error);
} finally {
loadingParent = false;
}
}
function getParentPreview(): string {
if (!parentEvent) {
return loadingParent ? 'Loading...' : 'Parent event not found';
}
// Create preview from parent (first 100 chars, plaintext with markdown stripped)
const plaintext = stripMarkdown(parentEvent.content);
return plaintext.slice(0, 100) + (plaintext.length > 100 ? '...' : '');
}
</script>
<div
class="reply-context mb-2 p-2 bg-fog-highlight dark:bg-fog-dark-highlight rounded text-xs text-fog-text-light dark:text-fog-dark-text-light"
>
<span class="font-semibold">Replying to:</span> {getParentPreview()}
{#if loadingParent}
<span class="text-xs opacity-70"> (loading...)</span>
{/if}
</div>
<style>
.reply-context {
border-left: 2px solid var(--fog-accent, #64748b);
}
:global(.dark) .reply-context {
border-left-color: var(--fog-dark-accent, #64748b);
}
:global(.highlight-parent) {
outline: 2px solid var(--fog-accent, #64748b);
outline-offset: 2px;
transition: outline 0.3s ease;
}
:global(.dark .highlight-parent) {
outline-color: var(--fog-dark-accent, #64748b);
}
</style>