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.
220 lines
7.0 KiB
220 lines
7.0 KiB
import { Button } from '@/components/ui/button' |
|
import { useToast } from '@/hooks/use-toast' |
|
import { createPictureNoteDraftEvent } from '@/lib/draft-event' |
|
import { cn } from '@/lib/utils' |
|
import { useNostr } from '@/providers/NostrProvider' |
|
import postContentCache from '@/services/post-content-cache.service' |
|
import { ChevronDown, Loader, LoaderCircle, Plus, X } from 'lucide-react' |
|
import { Dispatch, SetStateAction, useEffect, useRef, useState } from 'react' |
|
import { useTranslation } from 'react-i18next' |
|
import Image from '../Image' |
|
import TextareaWithMentions from '../TextareaWithMentions' |
|
import Mentions from './Mentions' |
|
import PostOptions from './PostOptions' |
|
import SendOnlyToSwitch from './SendOnlyToSwitch' |
|
import Uploader from './Uploader' |
|
import { preprocessContent } from './utils' |
|
|
|
export default function PicturePostContent({ close }: { close: () => void }) { |
|
const { t } = useTranslation() |
|
const { toast } = useToast() |
|
const { publish, checkLogin } = useNostr() |
|
const [content, setContent] = useState('') |
|
const [processedContent, setProcessedContent] = useState('') |
|
const [pictureInfos, setPictureInfos] = useState<{ url: string; tags: string[][] }[]>([]) |
|
const [posting, setPosting] = useState(false) |
|
const [showMoreOptions, setShowMoreOptions] = useState(false) |
|
const [addClientTag, setAddClientTag] = useState(false) |
|
const [mentions, setMentions] = useState<string[]>([]) |
|
const [specifiedRelayUrls, setSpecifiedRelayUrls] = useState<string[] | undefined>(undefined) |
|
const initializedRef = useRef(false) |
|
const canPost = !!content && !posting && pictureInfos.length > 0 |
|
|
|
useEffect(() => { |
|
const { content, pictureInfos } = postContentCache.getPicturePostCache() |
|
setContent(content) |
|
setPictureInfos(pictureInfos) |
|
setTimeout(() => { |
|
initializedRef.current = true |
|
}, 100) |
|
}, []) |
|
|
|
useEffect(() => { |
|
setProcessedContent(preprocessContent(content)) |
|
if (!initializedRef.current) return |
|
postContentCache.setPicturePostCache(content, pictureInfos) |
|
}, [content, pictureInfos]) |
|
|
|
const post = async (e: React.MouseEvent) => { |
|
e.stopPropagation() |
|
checkLogin(async () => { |
|
if (!canPost) { |
|
close() |
|
return |
|
} |
|
|
|
setPosting(true) |
|
try { |
|
if (!pictureInfos.length) { |
|
throw new Error(t('Picture note requires images')) |
|
} |
|
const draftEvent = await createPictureNoteDraftEvent( |
|
processedContent, |
|
pictureInfos, |
|
mentions, |
|
{ |
|
addClientTag, |
|
protectedEvent: !!specifiedRelayUrls |
|
} |
|
) |
|
await publish(draftEvent, { specifiedRelayUrls }) |
|
setContent('') |
|
setPictureInfos([]) |
|
close() |
|
} catch (error) { |
|
if (error instanceof AggregateError) { |
|
error.errors.forEach((e) => |
|
toast({ |
|
variant: 'destructive', |
|
title: t('Failed to post'), |
|
description: e.message |
|
}) |
|
) |
|
} else if (error instanceof Error) { |
|
toast({ |
|
variant: 'destructive', |
|
title: t('Failed to post'), |
|
description: error.message |
|
}) |
|
} |
|
console.error(error) |
|
return |
|
} finally { |
|
setPosting(false) |
|
} |
|
toast({ |
|
title: t('Post successful'), |
|
description: t('Your post has been published') |
|
}) |
|
}) |
|
} |
|
|
|
return ( |
|
<div className="space-y-4"> |
|
<div className="text-xs text-muted-foreground"> |
|
{t('A special note for picture-first clients like Olas')} |
|
</div> |
|
<PictureUploader pictureInfos={pictureInfos} setPictureInfos={setPictureInfos} /> |
|
<TextareaWithMentions |
|
className="h-32" |
|
setTextValue={setContent} |
|
textValue={content} |
|
placeholder={t('Write something...')} |
|
/> |
|
<SendOnlyToSwitch |
|
specifiedRelayUrls={specifiedRelayUrls} |
|
setSpecifiedRelayUrls={setSpecifiedRelayUrls} |
|
/> |
|
<div className="flex items-center justify-between"> |
|
<Button |
|
variant="link" |
|
className="text-foreground gap-0 px-0" |
|
onClick={() => setShowMoreOptions((pre) => !pre)} |
|
> |
|
{t('More options')} |
|
<ChevronDown className={`transition-transform ${showMoreOptions ? 'rotate-180' : ''}`} /> |
|
</Button> |
|
<div className="flex gap-2 items-center"> |
|
<Mentions content={processedContent} mentions={mentions} setMentions={setMentions} /> |
|
<div className="flex gap-2 items-center max-sm:hidden"> |
|
<Button |
|
variant="secondary" |
|
onClick={(e) => { |
|
e.stopPropagation() |
|
close() |
|
}} |
|
> |
|
{t('Cancel')} |
|
</Button> |
|
<Button type="submit" disabled={!canPost} onClick={post}> |
|
{posting && <LoaderCircle className="animate-spin" />} |
|
{t('Post')} |
|
</Button> |
|
</div> |
|
</div> |
|
</div> |
|
<PostOptions |
|
show={showMoreOptions} |
|
addClientTag={addClientTag} |
|
setAddClientTag={setAddClientTag} |
|
/> |
|
<div className="flex gap-2 items-center justify-around sm:hidden"> |
|
<Button |
|
className="w-full" |
|
variant="secondary" |
|
onClick={(e) => { |
|
e.stopPropagation() |
|
close() |
|
}} |
|
> |
|
{t('Cancel')} |
|
</Button> |
|
<Button className="w-full" type="submit" disabled={!canPost} onClick={post}> |
|
{posting && <LoaderCircle className="animate-spin" />} |
|
{t('Post')} |
|
</Button> |
|
</div> |
|
</div> |
|
) |
|
} |
|
|
|
function PictureUploader({ |
|
pictureInfos, |
|
setPictureInfos |
|
}: { |
|
pictureInfos: { url: string; tags: string[][] }[] |
|
setPictureInfos: Dispatch< |
|
SetStateAction< |
|
{ |
|
url: string |
|
tags: string[][] |
|
}[] |
|
> |
|
> |
|
}) { |
|
const [uploading, setUploading] = useState(false) |
|
|
|
return ( |
|
<div className="grid grid-cols-3 gap-4"> |
|
{pictureInfos.map(({ url }, index) => ( |
|
<div className="relative" key={`${index}-${url}`}> |
|
<Image image={{ url }} className="aspect-square w-full rounded-lg" /> |
|
<Button |
|
variant="destructive" |
|
className="absolute top-0 right-0 translate-x-1/2 -translate-y-1/2 rounded-full w-6 h-6 p-0" |
|
onClick={() => { |
|
setPictureInfos((prev) => prev.filter((_, i) => i !== index)) |
|
}} |
|
> |
|
<X /> |
|
</Button> |
|
</div> |
|
))} |
|
<Uploader |
|
onUploadSuccess={({ url, tags }) => { |
|
setPictureInfos((prev) => [...prev, { url, tags }]) |
|
}} |
|
onUploadingChange={setUploading} |
|
> |
|
<div |
|
className={cn( |
|
'flex flex-col gap-2 items-center justify-center aspect-square w-full rounded-lg border border-dashed', |
|
uploading ? 'cursor-not-allowed text-muted-foreground' : 'clickable' |
|
)} |
|
> |
|
{uploading ? <Loader size={36} className="animate-spin" /> : <Plus size={36} />} |
|
</div> |
|
</Uploader> |
|
</div> |
|
) |
|
}
|
|
|