import Note from '@/components/Note' import { Button } from '@/components/ui/button' import { ScrollArea } from '@/components/ui/scroll-area' import { createCommentDraftEvent, createPollDraftEvent, createShortTextNoteDraftEvent } from '@/lib/draft-event' 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, Settings, Smile, X } from 'lucide-react' import { Event, kinds } from 'nostr-tools' import { useEffect, useRef, useState } from 'react' import { useTranslation } from 'react-i18next' import { toast } from 'sonner' import EmojiPickerDialog from '../EmojiPickerDialog' import Mentions from './Mentions' import PollEditor from './PollEditor' import PostOptions from './PostOptions' import PostTextarea, { TPostTextareaHandle } from './PostTextarea' import SendOnlyToSwitch from './SendOnlyToSwitch' import Uploader from './Uploader' 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 [specifiedRelayUrls, setSpecifiedRelayUrls] = useState(undefined) const [mentions, setMentions] = useState([]) const [isNsfw, setIsNsfw] = useState(false) const [isPoll, setIsPoll] = useState(false) const [pollCreateData, setPollCreateData] = useState({ isMultipleChoice: false, options: ['', ''], endsAt: undefined, relays: [] }) const isFirstRender = useRef(true) const canPost = !!pubkey && !!text && !posting && !uploadProgresses.length && (!isPoll || pollCreateData.options.filter((option) => !!option.trim()).length >= 2) 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: [] } ) setSpecifiedRelayUrls(cachedSettings.specifiedRelayUrls) setAddClientTag(cachedSettings.addClientTag ?? false) } return } postEditorCache.setPostSettingsCache( { defaultContent, parentEvent }, { isNsfw, isPoll, pollCreateData, specifiedRelayUrls, addClientTag } ) }, [ defaultContent, parentEvent, isNsfw, isPoll, pollCreateData, specifiedRelayUrls, addClientTag ]) const post = async (e?: React.MouseEvent) => { e?.stopPropagation() checkLogin(async () => { if (!canPost) return setPosting(true) try { const draftEvent = parentEvent && parentEvent.kind !== kinds.ShortTextNote ? await createCommentDraftEvent(text, parentEvent, mentions, { addClientTag, protectedEvent: !!specifiedRelayUrls, isNsfw }) : isPoll ? await createPollDraftEvent(pubkey, text, mentions, pollCreateData, { addClientTag, isNsfw }) : await createShortTextNoteDraftEvent(text, mentions, { parentEvent, addClientTag, protectedEvent: !!specifiedRelayUrls, isNsfw }) const newEvent = await publish(draftEvent, { specifiedRelayUrls, additionalRelayUrls: isPoll ? pollCreateData.relays : [] }) addReplies([newEvent]) postEditorCache.clearPostCache({ defaultContent, parentEvent }) close() } catch (error) { if (error instanceof AggregateError) { error.errors.forEach((e) => toast.error(`${t('Failed to post')}: ${e.message}`)) } else if (error instanceof Error) { toast.error(`${t('Failed to post')}: ${error.message}`) } console.error(error) return } finally { setPosting(false) } toast.success(t('Post successful'), { duration: 2000 }) }) } const handlePollToggle = () => { if (parentEvent) return setIsPoll((prev) => !prev) } 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 (
{parentEvent && (
)} post()} className={isPoll ? 'min-h-20' : 'min-h-52'} onUploadStart={handleUploadStart} onUploadProgress={handleUploadProgress} onUploadEnd={handleUploadEnd} /> {isPoll && ( )} {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 && ( )}
) }