import Note from '@/components/Note' import { Button } from '@/components/ui/button' import { ScrollArea } from '@/components/ui/scroll-area' import { createCommentDraftEvent, createPollDraftEvent, createPublicMessageDraftEvent, createPublicMessageReplyDraftEvent, createShortTextNoteDraftEvent, deleteDraftEventCache } from '@/lib/draft-event' import { ExtendedKind } from '@/constants' import { isTouchDevice } from '@/lib/utils' import { useNostr } from '@/providers/NostrProvider' import { useReply } from '@/providers/ReplyProvider' import postEditorCache from '@/services/post-editor-cache.service' import { TPollCreateData } from '@/types' import { ImageUp, ListTodo, LoaderCircle, MessageCircle, Settings, Smile, X } from 'lucide-react' import { Event, kinds } from 'nostr-tools' import { useCallback, useEffect, useMemo, useRef, useState } from 'react' import { useTranslation } from 'react-i18next' import { toast } from 'sonner' import EmojiPickerDialog from '../EmojiPickerDialog' import Mentions, { extractMentions } from './Mentions' import PollEditor from './PollEditor' import PostOptions from './PostOptions' import PostRelaySelector from './PostRelaySelector' import PostTextarea, { TPostTextareaHandle } from './PostTextarea' import Uploader from './Uploader' import RelayStatusDisplay from '@/components/RelayStatusDisplay' export default function PostContent({ defaultContent = '', parentEvent, close, openFrom }: { defaultContent?: string parentEvent?: Event close: () => void openFrom?: string[] }) { const { t } = useTranslation() const { pubkey, publish, checkLogin } = useNostr() const { addReplies } = useReply() const [text, setText] = useState('') const textareaRef = useRef(null) const [posting, setPosting] = useState(false) const [uploadProgresses, setUploadProgresses] = useState< { file: File; progress: number; cancel: () => void }[] >([]) const [showMoreOptions, setShowMoreOptions] = useState(false) const [addClientTag, setAddClientTag] = useState(false) const [mentions, setMentions] = useState([]) const [isNsfw, setIsNsfw] = useState(false) const [isPoll, setIsPoll] = useState(false) const [isPublicMessage, setIsPublicMessage] = useState(false) const [publicMessageRecipients, setPublicMessageRecipients] = useState([]) const [isProtectedEvent, setIsProtectedEvent] = useState(false) const [additionalRelayUrls, setAdditionalRelayUrls] = useState([]) const [pollCreateData, setPollCreateData] = useState({ isMultipleChoice: false, options: ['', ''], endsAt: undefined, relays: [] }) const [minPow, setMinPow] = useState(0) const [relayStatuses, setRelayStatuses] = useState>([]) const [showRelayStatus, setShowRelayStatus] = useState(false) const [lastPublishedEvent, setLastPublishedEvent] = useState(null) const isFirstRender = useRef(true) const canPost = useMemo(() => { const result = ( !!pubkey && !!text && !posting && !uploadProgresses.length && (!isPoll || pollCreateData.options.filter((option) => !!option.trim()).length >= 2) && (!isPublicMessage || publicMessageRecipients.length > 0 || parentEvent?.kind === ExtendedKind.PUBLIC_MESSAGE) && (!isProtectedEvent || additionalRelayUrls.length > 0) ) // Debug logging for public message replies if (parentEvent?.kind === ExtendedKind.PUBLIC_MESSAGE) { console.log('Public message reply debug:', { pubkey: !!pubkey, text: !!text, posting, uploadProgresses: uploadProgresses.length, isPoll, pollCreateDataValid: !isPoll || pollCreateData.options.filter((option) => !!option.trim()).length >= 2, publicMessageCheck: !isPublicMessage || publicMessageRecipients.length > 0 || parentEvent?.kind === ExtendedKind.PUBLIC_MESSAGE, protectedEventCheck: !isProtectedEvent || additionalRelayUrls.length > 0, canPost: result }) } return result }, [ pubkey, text, posting, uploadProgresses, isPoll, pollCreateData, isPublicMessage, publicMessageRecipients, parentEvent?.kind, isProtectedEvent, additionalRelayUrls ]) useEffect(() => { if (isFirstRender.current) { isFirstRender.current = false const cachedSettings = postEditorCache.getPostSettingsCache({ defaultContent, parentEvent }) if (cachedSettings) { setIsNsfw(cachedSettings.isNsfw ?? false) setIsPoll(cachedSettings.isPoll ?? false) setPollCreateData( cachedSettings.pollCreateData ?? { isMultipleChoice: false, options: ['', ''], endsAt: undefined, relays: [] } ) setAddClientTag(cachedSettings.addClientTag ?? false) } return } postEditorCache.setPostSettingsCache( { defaultContent, parentEvent }, { isNsfw, isPoll, pollCreateData, addClientTag } ) }, [defaultContent, parentEvent, isNsfw, isPoll, pollCreateData, addClientTag]) // Extract mentions from content for public messages const extractMentionsFromContent = useCallback(async (content: string) => { try { // First try to extract nostr: protocol mentions const { pubkeys: nostrPubkeys } = await extractMentions(content, undefined) // Also extract regular @ mentions (simple pattern for now) const atMentions = content.match(/@[a-zA-Z0-9_]+/g) || [] console.log('Nostr mentions:', nostrPubkeys) console.log('@ mentions:', atMentions) // For now, we'll use the nostr mentions and show that we detected @ mentions // In a real implementation, you'd resolve @ mentions to pubkeys setPublicMessageRecipients(nostrPubkeys) } catch (error) { console.error('Error extracting mentions:', error) setPublicMessageRecipients([]) } }, []) useEffect(() => { if (!isPublicMessage) { setPublicMessageRecipients([]) return } if (!text) { setPublicMessageRecipients([]) return } // Debounce the mention extraction const timeoutId = setTimeout(() => { extractMentionsFromContent(text) }, 300) return () => { clearTimeout(timeoutId) } }, [text, isPublicMessage, extractMentionsFromContent]) const post = async (e?: React.MouseEvent) => { e?.stopPropagation() checkLogin(async () => { if (!canPost) { console.log('❌ Cannot post - canPost is false') return } // console.log('🚀 Starting post process:', { // isPublicMessage, // parentEventKind: parentEvent?.kind, // parentEventId: parentEvent?.id, // text: text.substring(0, 50) + '...', // mentions: mentions.length, // canPost // }) setPosting(true) try { let draftEvent if (isPublicMessage) { draftEvent = await createPublicMessageDraftEvent(text, publicMessageRecipients, { addClientTag, isNsfw }) } else if (parentEvent && parentEvent.kind === ExtendedKind.PUBLIC_MESSAGE) { draftEvent = await createPublicMessageReplyDraftEvent(text, parentEvent, mentions, { addClientTag, isNsfw }) } else if (parentEvent && parentEvent.kind !== kinds.ShortTextNote) { draftEvent = await createCommentDraftEvent(text, parentEvent, mentions, { addClientTag, protectedEvent: isProtectedEvent, isNsfw }) } else if (isPoll) { draftEvent = await createPollDraftEvent(pubkey!, text, mentions, pollCreateData, { addClientTag, isNsfw }) } else { draftEvent = await createShortTextNoteDraftEvent(text, mentions, { parentEvent, addClientTag, protectedEvent: isProtectedEvent, isNsfw }) } // console.log('Publishing draft event:', draftEvent) const newEvent = await publish(draftEvent, { specifiedRelayUrls: isProtectedEvent ? additionalRelayUrls : undefined, additionalRelayUrls: isPoll ? pollCreateData.relays : additionalRelayUrls, minPow }) // console.log('Published event:', newEvent) // Check if we have relay status information console.log('Published event:', newEvent) console.log('Relay statuses:', (newEvent as any).relayStatuses) if ((newEvent as any).relayStatuses) { setRelayStatuses((newEvent as any).relayStatuses) setLastPublishedEvent(newEvent) setShowRelayStatus(true) // Show success message with relay count const successCount = (newEvent as any).relayStatuses.filter((s: any) => s.success).length const totalCount = (newEvent as any).relayStatuses.length toast.success(t('Post successful - published to {{count}} of {{total}} relays', { count: successCount, total: totalCount }), { duration: 4000 }) // Don't close immediately if we have relay status to show setTimeout(() => { postEditorCache.clearPostCache({ defaultContent, parentEvent }) deleteDraftEventCache(draftEvent) addReplies([newEvent]) close() }, 8000) // Give user more time to see the relay status } else { toast.success(t('Post successful'), { duration: 2000 }) postEditorCache.clearPostCache({ defaultContent, parentEvent }) deleteDraftEventCache(draftEvent) addReplies([newEvent]) close() } } catch (error) { console.error('Publishing error:', error) // Handle different types of errors with user-friendly messages let errorMessage = t('Failed to post') if (error instanceof Error) { if (error.message.includes('timeout')) { errorMessage = t('Posting timed out. Your post may have been published to some relays.') } else if (error.message.includes('auth-required') || error.message.includes('auth required')) { errorMessage = t('Some relays require authentication. Please try again or use different relays.') } else if (error.message.includes('blocked')) { errorMessage = t('You are blocked from posting to some relays.') } else if (error.message.includes('rate limit')) { errorMessage = t('Rate limited. Please wait before trying again.') } else if (error.message.includes('writes disabled')) { errorMessage = t('Some relays have temporarily disabled writes.') } else { errorMessage = `${t('Failed to post')}: ${error.message}` } } else if (error instanceof AggregateError) { // Handle multiple relay failures const hasAuthErrors = error.errors.some(err => err instanceof Error && err.message.includes('auth-required') ) const hasBlockedErrors = error.errors.some(err => err instanceof Error && err.message.includes('blocked') ) const hasWriteDisabledErrors = error.errors.some(err => err instanceof Error && err.message.includes('writes disabled') ) if (hasAuthErrors) { errorMessage = t('Some relays require authentication. Your post may have been published to other relays.') } else if (hasBlockedErrors) { errorMessage = t('You are blocked from some relays. Your post may have been published to other relays.') } else if (hasWriteDisabledErrors) { errorMessage = t('Some relays have disabled writes. Your post may have been published to other relays.') } else { errorMessage = t('Failed to publish to some relays. Your post may have been published to other relays.') } } toast.error(errorMessage, { duration: 8000 }) return } finally { setPosting(false) } }) } const handlePollToggle = () => { if (parentEvent) return setIsPoll((prev) => !prev) } const handlePublicMessageToggle = () => { if (parentEvent) return setIsPublicMessage((prev) => !prev) if (!isPublicMessage) { // When enabling public message mode, clear other modes setIsPoll(false) } } const handleUploadStart = (file: File, cancel: () => void) => { setUploadProgresses((prev) => [...prev, { file, progress: 0, cancel }]) } const handleUploadProgress = (file: File, progress: number) => { setUploadProgresses((prev) => prev.map((item) => (item.file === file ? { ...item, progress } : item)) ) } const handleUploadEnd = (file: File) => { setUploadProgresses((prev) => prev.filter((item) => item.file !== file)) } return (
{/* Dynamic Title based on mode */}
{parentEvent ? (
{parentEvent.kind === ExtendedKind.PUBLIC_MESSAGE ? t('Reply to Public Message') : t('Reply to') }
) : isPoll ? ( t('New Poll') ) : isPublicMessage ? ( t('New Public Message') ) : ( t('New Note') )}
{parentEvent && (
)} post()} className={isPoll ? 'min-h-20' : 'min-h-52'} onUploadStart={handleUploadStart} onUploadProgress={handleUploadProgress} onUploadEnd={handleUploadEnd} kind={isPublicMessage ? ExtendedKind.PUBLIC_MESSAGE : isPoll ? ExtendedKind.POLL : kinds.ShortTextNote} /> {isPoll && ( )} {isPublicMessage && (
{t('Recipients')}
{publicMessageRecipients.length > 0 ? (
{t('Recipients detected from your message:')} {publicMessageRecipients.length}
) : (
{t('Add recipients using nostr: mentions (e.g., nostr:npub1...) or the recipient selector above')}
)}
)} {uploadProgresses.length > 0 && uploadProgresses.map(({ file, progress, cancel }, index) => (
{file.name ?? t('Uploading...')}
))} {!isPoll && ( )}
{ textareaRef.current?.appendText(url, true) }} onUploadStart={handleUploadStart} onUploadEnd={handleUploadEnd} onProgress={handleUploadProgress} accept="image/*,video/*,audio/*" > {/* I'm not sure why, but after triggering the virtual keyboard, opening the emoji picker drawer causes an issue, the emoji I tap isn't the one that gets inserted. */} {!isTouchDevice() && ( { if (!emoji) return textareaRef.current?.insertEmoji(emoji) }} > )} {!parentEvent && ( )} {!parentEvent && ( )}
{showRelayStatus && relayStatuses.length > 0 && (

📡 Publishing Results

Your post has been published. Here's the status for each relay:

s.success).length} totalCount={relayStatuses.length} />
This dialog will close automatically in a few seconds
)}
) }