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.
136 lines
4.4 KiB
136 lines
4.4 KiB
import { Button } from '@/components/ui/button' |
|
import { DialogFooter } from '@/components/ui/dialog' |
|
import { Textarea } from '@/components/ui/textarea' |
|
import { ExtendedKind } from '@/constants' |
|
import { createPublicMessageDraftEvent } from '@/lib/draft-event' |
|
import { createFakeEvent } from '@/lib/event' |
|
import { LoginRequiredError } from '@/lib/nostr-errors' |
|
import { pubkeyToNpub } from '@/lib/pubkey' |
|
import { showSimplePublishSuccess } from '@/lib/publishing-feedback' |
|
import { cn } from '@/lib/utils' |
|
import { useNostr } from '@/providers/NostrProvider' |
|
import { useEffect, useMemo, useRef, useState } from 'react' |
|
import { useTranslation } from 'react-i18next' |
|
import { toast } from 'sonner' |
|
import MarkdownArticle from '../Note/MarkdownArticle/MarkdownArticle' |
|
|
|
const TIP_NOTICE_DEFAULT_KEY = 'I just sent you a tip!' |
|
|
|
function defaultTipNoticeMessage(recipientPubkey: string, tipText: string): string { |
|
const npub = pubkeyToNpub(recipientPubkey) |
|
return `nostr:${npub} ${tipText}` |
|
} |
|
|
|
export default function PublicMessageForm({ |
|
recipientPubkey, |
|
onBack, |
|
onDone |
|
}: { |
|
recipientPubkey: string |
|
onBack: () => void |
|
onDone: () => void |
|
}) { |
|
const { t } = useTranslation() |
|
const { publish, checkLogin, pubkey: selfPubkey } = useNostr() |
|
const [sending, setSending] = useState(false) |
|
const [message, setMessage] = useState('') |
|
const textareaRef = useRef<HTMLTextAreaElement>(null) |
|
|
|
const tipText = t(TIP_NOTICE_DEFAULT_KEY) |
|
|
|
useEffect(() => { |
|
setMessage(defaultTipNoticeMessage(recipientPubkey, tipText)) |
|
}, [recipientPubkey, tipText]) |
|
|
|
useEffect(() => { |
|
const id = requestAnimationFrame(() => { |
|
textareaRef.current?.focus() |
|
textareaRef.current?.setSelectionRange( |
|
textareaRef.current.value.length, |
|
textareaRef.current.value.length |
|
) |
|
}) |
|
return () => cancelAnimationFrame(id) |
|
}, []) |
|
|
|
const previewEvent = useMemo(() => { |
|
return createFakeEvent({ |
|
kind: ExtendedKind.PUBLIC_MESSAGE, |
|
pubkey: selfPubkey ?? '', |
|
content: message, |
|
tags: [['p', recipientPubkey]] |
|
}) |
|
}, [message, recipientPubkey, selfPubkey]) |
|
|
|
const handleSend = () => { |
|
const trimmed = message.trim() |
|
if (!trimmed) return |
|
checkLogin(async () => { |
|
if (selfPubkey === recipientPubkey) { |
|
onDone() |
|
return |
|
} |
|
setSending(true) |
|
try { |
|
const draft = await createPublicMessageDraftEvent(trimmed, [recipientPubkey], { |
|
addClientTag: true |
|
}) |
|
await publish(draft, { disableFallbacks: true }) |
|
showSimplePublishSuccess(t('Tip notice sent')) |
|
onDone() |
|
} catch (error) { |
|
if (error instanceof LoginRequiredError) return |
|
toast.error( |
|
t('Failed to send tip notice', { |
|
error: error instanceof Error ? error.message : String(error) |
|
}) |
|
) |
|
} finally { |
|
setSending(false) |
|
} |
|
}) |
|
} |
|
|
|
return ( |
|
<div className="min-w-0"> |
|
<p className="text-sm text-muted-foreground">{t('Tip notice prompt description')}</p> |
|
<div className="mt-3 min-w-0 max-w-full"> |
|
<Textarea |
|
ref={textareaRef} |
|
value={message} |
|
onChange={(e) => setMessage(e.target.value)} |
|
disabled={sending} |
|
rows={6} |
|
className="min-h-[10rem] w-full max-w-full resize-y box-border text-sm leading-relaxed focus-visible:ring-inset" |
|
aria-label={t('Tip notice prompt description')} |
|
/> |
|
</div> |
|
{previewEvent ? ( |
|
<div className="mt-4 min-w-0"> |
|
<p className="text-xs font-medium text-muted-foreground">{t('Preview')}</p> |
|
<div |
|
className={cn( |
|
'mt-1.5 max-h-56 min-w-0 overflow-y-auto overflow-x-hidden rounded-md border border-border', |
|
'bg-muted/25 px-3 py-2' |
|
)} |
|
> |
|
<MarkdownArticle event={previewEvent} hideMetadata lazyMedia={false} className="text-sm" /> |
|
</div> |
|
</div> |
|
) : null} |
|
<DialogFooter className="mt-4 flex w-full min-w-0 flex-col-reverse gap-2 sm:flex-row sm:justify-end"> |
|
<Button type="button" variant="outline" className="w-full min-w-0 sm:w-auto" onClick={onBack} disabled={sending}> |
|
{t('Back')} |
|
</Button> |
|
<Button |
|
type="button" |
|
className="w-full min-w-0 sm:w-auto" |
|
onClick={handleSend} |
|
disabled={sending || !message.trim()} |
|
> |
|
{t('Send')} |
|
</Button> |
|
</DialogFooter> |
|
</div> |
|
) |
|
}
|
|
|