diff --git a/src/classes/nostr-post-action.ts b/src/classes/nostr-post-action.ts index c23309e18..3fc031bb4 100644 --- a/src/classes/nostr-post-action.ts +++ b/src/classes/nostr-post-action.ts @@ -2,7 +2,7 @@ import { Subject, Subscription } from "rxjs"; import { relayPool } from "../services/relays"; import { NostrEvent } from "../types/nostr-event"; -type PostResult = { url: string; message?: string; status: boolean }; +export type PostResult = { url: string; message?: string; status: boolean }; export function nostrPostAction(relays: string[], event: NostrEvent, timeout: number = 5000) { const subject = new Subject(); diff --git a/src/components/icons.tsx b/src/components/icons.tsx index aabbd83ea..b0afaf828 100644 --- a/src/components/icons.tsx +++ b/src/components/icons.tsx @@ -133,3 +133,9 @@ export const ShareIcon = createIcon({ d: "M13.12 17.023l-4.199-2.29a4 4 0 1 1 0-5.465l4.2-2.29a4 4 0 1 1 .959 1.755l-4.2 2.29a4.008 4.008 0 0 1 0 1.954l4.199 2.29a4 4 0 1 1-.959 1.755zM6 14a2 2 0 1 0 0-4 2 2 0 0 0 0 4zm11-6a2 2 0 1 0 0-4 2 2 0 0 0 0 4zm0 12a2 2 0 1 0 0-4 2 2 0 0 0 0 4z", defaultProps, }); + +export const ReplyIcon = createIcon({ + displayName: "reply-icon", + d: "M11 20L1 12l10-8v5c5.523 0 10 4.477 10 10 0 .273-.01.543-.032.81-1.463-2.774-4.33-4.691-7.655-4.805L13 15h-2v5zm-2-7h4.034l.347.007c1.285.043 2.524.31 3.676.766C15.59 12.075 13.42 11 11 11H9V8.161L4.202 12 9 15.839V13z", + defaultProps, +}); diff --git a/src/components/inline-new-post.tsx b/src/components/inline-new-post.tsx deleted file mode 100644 index ed05ccdb7..000000000 --- a/src/components/inline-new-post.tsx +++ /dev/null @@ -1,48 +0,0 @@ -import { Button, useDisclosure } from "@chakra-ui/react"; -import moment from "moment"; -import { useState } from "react"; -import { lastValueFrom } from "rxjs"; -import { nostrPostAction } from "../classes/nostr-post-action"; -import settings from "../services/settings"; -import { DraftNostrEvent } from "../types/nostr-event"; -import { AddIcon } from "./icons"; -import { PostForm, PostFormValues } from "./post-modal/post-form"; - -export type InlineNewPostProps = {}; - -export const InlineNewPost = ({}: InlineNewPostProps) => { - const { isOpen, onClose, onOpen } = useDisclosure(); - const [loading, setLoading] = useState(false); - - const handlePostSubmit = async (values: PostFormValues) => { - setLoading(true); - const draft: DraftNostrEvent = { - content: values.content, - tags: [], - kind: 1, - created_at: moment().unix(), - }; - - if (window.nostr) { - const event = await window.nostr.signEvent(draft); - - const postResults = nostrPostAction(settings.relays.value, event); - - postResults.subscribe((result) => { - console.log(`Posted event to ${result.url}: ${result.message}`); - }); - - await lastValueFrom(postResults); - } - setLoading(false); - }; - - if (isOpen) { - return ; - } - return ( - - ); -}; diff --git a/src/components/note-link.tsx b/src/components/note-link.tsx index 9861dfc8b..ff534c587 100644 --- a/src/components/note-link.tsx +++ b/src/components/note-link.tsx @@ -11,7 +11,7 @@ export const NoteLink = ({ noteId, ...props }: NoteLinkProps) => { const note1 = normalizeToBech32(noteId, Bech32Prefix.Note) ?? noteId; return ( - + {truncatedId(note1)} ); diff --git a/src/components/note/index.tsx b/src/components/note/index.tsx index f8ba3122a..9e1ecd162 100644 --- a/src/components/note/index.tsx +++ b/src/components/note/index.tsx @@ -1,7 +1,7 @@ -import React from "react"; +import React, { useContext } from "react"; import { Link as RouterLink } from "react-router-dom"; import moment from "moment"; -import { Box, Card, CardBody, CardFooter, CardHeader, Flex, Heading, Link } from "@chakra-ui/react"; +import { Box, Card, CardBody, CardFooter, CardHeader, Flex, Heading, IconButton, Link } from "@chakra-ui/react"; import { NostrEvent } from "../../types/nostr-event"; import { UserAvatarLink } from "../user-avatar-link"; import { Bech32Prefix, normalizeToBech32 } from "../../helpers/nip-19"; @@ -15,20 +15,26 @@ import { UserTipButton } from "../user-tip-button"; import { NoteRelays } from "./note-relays"; import { useIsMobile } from "../../hooks/use-is-mobile"; import { UserLink } from "../user-link"; +import { ReplyIcon } from "../icons"; +import { PostModalContext } from "../../providers/post-modal-provider"; +import { buildReply } from "../../helpers/nostr-event"; export type NoteProps = { event: NostrEvent; }; export const Note = React.memo(({ event }: NoteProps) => { const isMobile = useIsMobile(); + const { openModal } = useContext(PostModalContext); const pubkey = useSubject(identity.pubkey); const contacts = useUserContacts(pubkey); const following = contacts?.contacts || []; + const reply = () => openModal(buildReply(event)); + return ( - - + + @@ -40,12 +46,12 @@ export const Note = React.memo(({ event }: NoteProps) => { - - - - + + - + + } title="Reply" aria-label="Reply" onClick={reply} size="xs" /> + diff --git a/src/components/note/note-contents.tsx b/src/components/note/note-contents.tsx index f0ed5d98c..beae86b04 100644 --- a/src/components/note/note-contents.tsx +++ b/src/components/note/note-contents.tsx @@ -21,7 +21,11 @@ import settings from "../../services/settings"; const BlurredImage = (props: ImageProps) => { const { isOpen, onToggle } = useDisclosure(); - return ; + return ( + + + + ); }; type EmbedType = { diff --git a/src/components/page.tsx b/src/components/page.tsx index 8519e305d..d73525e84 100644 --- a/src/components/page.tsx +++ b/src/components/page.tsx @@ -9,6 +9,7 @@ import { useIsMobile } from "../hooks/use-is-mobile"; import identity from "../services/identity"; import { FollowingList } from "./following-list"; import { ReloadPrompt } from "./reload-prompt"; +import { PostModalProvider } from "../providers/post-modal-provider"; export const Page = ({ children }: { children: React.ReactNode }) => { const navigate = useNavigate(); @@ -19,7 +20,9 @@ export const Page = ({ children }: { children: React.ReactNode }) => { - {children} + + {children} + } aria-label="Home" onClick={() => navigate("/")} flexGrow="1" size="lg" /> @@ -72,7 +75,9 @@ export const Page = ({ children }: { children: React.ReactNode }) => { - {children} + + {children} + Following diff --git a/src/components/post-modal/index.tsx b/src/components/post-modal/index.tsx index a143f4991..b0a9a3da0 100644 --- a/src/components/post-modal/index.tsx +++ b/src/components/post-modal/index.tsx @@ -1,20 +1,111 @@ -import { Modal, ModalOverlay, ModalContent, ModalHeader, ModalBody, ModalCloseButton } from "@chakra-ui/react"; -import { PostForm, PostFormProps } from "./post-form"; +import { + Modal, + ModalOverlay, + ModalContent, + ModalBody, + ModalFooter, + Flex, + Button, + Textarea, + Text, +} from "@chakra-ui/react"; +import moment from "moment"; +import React, { useState } from "react"; +import { useList } from "react-use"; +import { nostrPostAction, PostResult } from "../../classes/nostr-post-action"; +import { getReferences } from "../../helpers/nostr-event"; +import { useIsMobile } from "../../hooks/use-is-mobile"; +import settings from "../../services/settings"; +import { DraftNostrEvent, NostrEvent } from "../../types/nostr-event"; +import { NoteLink } from "../note-link"; +import { PostResults } from "./post-results"; -type PostModalProps = PostFormProps & { +function emptyDraft(): DraftNostrEvent { + return { + content: "", + kind: 1, + tags: [], + created_at: moment().unix(), + }; +} + +type PostModalProps = { isOpen: boolean; onClose: () => void; + initialDraft?: Partial; }; -export const PostModal = ({ isOpen, onClose, onSubmit, onCancel }: PostModalProps) => ( - - - - New Post - - - - - - -); +export const PostModal = ({ isOpen, onClose, initialDraft }: PostModalProps) => { + const isMobile = useIsMobile(); + const pad = isMobile ? "2" : "4"; + + const [waiting, setWaiting] = useState(false); + const [signedEvent, setSignedEvent] = useState(null); + const [results, resultsActions] = useList(); + const [draft, setDraft] = useState(() => Object.assign(emptyDraft(), initialDraft)); + + const handleContentChange: React.ChangeEventHandler = (event) => { + setDraft((d) => ({ ...d, content: event.target.value })); + }; + + const handleSubmit = async () => { + if (window.nostr) { + setWaiting(true); + const updatedDraft: DraftNostrEvent = { ...draft, created_at: moment().unix() }; + const event = await window.nostr.signEvent(updatedDraft); + setWaiting(false); + setSignedEvent(event); + + const postResults = nostrPostAction(settings.relays.value, event); + + postResults.subscribe({ + next(result) { + resultsActions.push(result); + }, + }); + } + }; + + const refs = getReferences(draft); + + const canSubmit = draft.content.length > 0; + + const renderContent = () => { + if (signedEvent) { + return ( + + + + ); + } + return ( + <> + + {refs.replyId && ( + + Replying to: + + )} +