From 9cd5562965b68f8721c44d7f741804a4d54bf56d Mon Sep 17 00:00:00 2001 From: Herman Wikner Date: Thu, 31 Aug 2023 16:37:37 +0200 Subject: [PATCH] feat: comments wip --- .../fieldActions/commentFieldAction.tsx | 22 -- dev/test-studio/sanity.config.ts | 3 +- packages/sanity/package.json | 1 + .../__workshop__/CommentInputStory.tsx | 42 ++++ .../__workshop__/CommentsListStory.tsx | 112 +++++++++ .../__workshop__/MentionOptionsHookStory.tsx | 21 ++ .../__workshop__/MentionsMenuStory.tsx | 22 ++ .../src/core/comments/__workshop__/index.ts | 29 +++ .../components/CommentDeleteDialog.tsx | 70 ++++++ .../src/core/comments/components/constants.ts | 4 + .../comments/components/icons/MentionIcon.tsx | 27 ++ .../comments/components/icons/SendIcon.tsx | 27 ++ .../core/comments/components/icons/index.ts | 2 + .../src/core/comments/components/index.ts | 3 + .../comments/components/list/CommentsList.tsx | 85 +++++++ .../components/list/CommentsListItem.tsx | 98 ++++++++ .../list/CommentsListItemLayout.tsx | 230 ++++++++++++++++++ .../core/comments/components/list/index.ts | 2 + .../components/mentions/MentionsMenu.tsx | 114 +++++++++ .../components/mentions/MentionsMenuItem.tsx | 35 +++ .../comments/components/mentions/index.ts | 1 + .../comments/components/pte/Serializer.tsx | 43 ++++ .../pte/blocks/MentionInlineBlock.tsx | 50 ++++ .../components/pte/blocks/NormalBlock.tsx | 21 ++ .../comments/components/pte/blocks/index.ts | 2 + .../pte/comment-input/CommentInput.tsx | 86 +++++++ .../pte/comment-input/CommentInputInner.tsx | 131 ++++++++++ .../comment-input/CommentInputProvider.tsx | 203 ++++++++++++++++ .../components/pte/comment-input/Editable.tsx | 152 ++++++++++++ .../components/pte/comment-input/index.ts | 3 + .../pte/comment-input/useCommentInput.ts | 12 + .../core/comments/components/pte/config.ts | 33 +++ .../src/core/comments/components/pte/index.ts | 2 + .../comments/components/pte/render/index.ts | 2 + .../components/pte/render/renderBlock.tsx | 9 + .../components/pte/render/renderChild.tsx | 15 ++ .../core/comments/context/CommentsContext.ts | 4 + .../comments/context/CommentsProvider.tsx | 112 +++++++++ .../sanity/src/core/comments/context/index.ts | 2 + packages/sanity/src/core/comments/helpers.ts | 23 ++ .../sanity/src/core/comments/hooks/index.ts | 5 + .../comments/hooks/useCommentOperations.ts | 186 ++++++++++++++ .../src/core/comments/hooks/useComments.ts | 17 ++ .../core/comments/hooks/useCommentsClient.ts | 21 ++ .../comments/hooks/useFieldCommentsCount.ts | 18 ++ .../core/comments/hooks/useMentionOptions.ts | 94 +++++++ packages/sanity/src/core/comments/index.ts | 18 ++ .../sanity/src/core/comments/store/index.ts | 1 + .../sanity/src/core/comments/store/reducer.ts | 106 ++++++++ .../core/comments/store/useCommentsStore.ts | 125 ++++++++++ packages/sanity/src/core/comments/types.ts | 129 ++++++++++ .../form/components/formField/FormField.tsx | 6 +- .../formField/FormFieldBaseHeader.tsx | 86 ++++--- .../components/formField/FormFieldSet.tsx | 7 +- .../src/core/form/hooks/useDidUpdate.ts | 5 + packages/sanity/src/core/form/index.ts | 1 + .../inputs/ReferenceInput/ReferenceField.tsx | 5 +- .../studio/inputResolver/fieldResolver.tsx | 3 + .../sanity/src/core/form/types/fieldProps.ts | 9 + packages/sanity/src/core/index.ts | 2 + .../src/core/store/_legacy/project/types.ts | 2 + .../src/desk/comments/field/CommentField.tsx | 72 ++++++ .../comments/field/CommentFieldButton.tsx | 178 ++++++++++++++ .../sanity/src/desk/comments/field/index.ts | 1 + packages/sanity/src/desk/comments/index.ts | 1 + .../comments/inspector/CommentsInspector.tsx | 141 +++++++++++ .../src/desk/comments/inspector/index.ts | 29 +++ packages/sanity/src/desk/comments/plugin.ts | 15 ++ packages/sanity/src/desk/deskTool.ts | 6 +- .../panes/document/DocumentPaneProvider.tsx | 31 +-- .../src/desk/panes/document/constants.ts | 1 + 71 files changed, 3099 insertions(+), 77 deletions(-) delete mode 100644 dev/test-studio/fieldActions/commentFieldAction.tsx create mode 100644 packages/sanity/src/core/comments/__workshop__/CommentInputStory.tsx create mode 100644 packages/sanity/src/core/comments/__workshop__/CommentsListStory.tsx create mode 100644 packages/sanity/src/core/comments/__workshop__/MentionOptionsHookStory.tsx create mode 100644 packages/sanity/src/core/comments/__workshop__/MentionsMenuStory.tsx create mode 100644 packages/sanity/src/core/comments/__workshop__/index.ts create mode 100644 packages/sanity/src/core/comments/components/CommentDeleteDialog.tsx create mode 100644 packages/sanity/src/core/comments/components/constants.ts create mode 100644 packages/sanity/src/core/comments/components/icons/MentionIcon.tsx create mode 100644 packages/sanity/src/core/comments/components/icons/SendIcon.tsx create mode 100644 packages/sanity/src/core/comments/components/icons/index.ts create mode 100644 packages/sanity/src/core/comments/components/index.ts create mode 100644 packages/sanity/src/core/comments/components/list/CommentsList.tsx create mode 100644 packages/sanity/src/core/comments/components/list/CommentsListItem.tsx create mode 100644 packages/sanity/src/core/comments/components/list/CommentsListItemLayout.tsx create mode 100644 packages/sanity/src/core/comments/components/list/index.ts create mode 100644 packages/sanity/src/core/comments/components/mentions/MentionsMenu.tsx create mode 100644 packages/sanity/src/core/comments/components/mentions/MentionsMenuItem.tsx create mode 100644 packages/sanity/src/core/comments/components/mentions/index.ts create mode 100644 packages/sanity/src/core/comments/components/pte/Serializer.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/blocks/MentionInlineBlock.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/blocks/NormalBlock.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/blocks/index.ts create mode 100644 packages/sanity/src/core/comments/components/pte/comment-input/CommentInput.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/comment-input/CommentInputInner.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/comment-input/CommentInputProvider.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/comment-input/Editable.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/comment-input/index.ts create mode 100644 packages/sanity/src/core/comments/components/pte/comment-input/useCommentInput.ts create mode 100644 packages/sanity/src/core/comments/components/pte/config.ts create mode 100644 packages/sanity/src/core/comments/components/pte/index.ts create mode 100644 packages/sanity/src/core/comments/components/pte/render/index.ts create mode 100644 packages/sanity/src/core/comments/components/pte/render/renderBlock.tsx create mode 100644 packages/sanity/src/core/comments/components/pte/render/renderChild.tsx create mode 100644 packages/sanity/src/core/comments/context/CommentsContext.ts create mode 100644 packages/sanity/src/core/comments/context/CommentsProvider.tsx create mode 100644 packages/sanity/src/core/comments/context/index.ts create mode 100644 packages/sanity/src/core/comments/helpers.ts create mode 100644 packages/sanity/src/core/comments/hooks/index.ts create mode 100644 packages/sanity/src/core/comments/hooks/useCommentOperations.ts create mode 100644 packages/sanity/src/core/comments/hooks/useComments.ts create mode 100644 packages/sanity/src/core/comments/hooks/useCommentsClient.ts create mode 100644 packages/sanity/src/core/comments/hooks/useFieldCommentsCount.ts create mode 100644 packages/sanity/src/core/comments/hooks/useMentionOptions.ts create mode 100644 packages/sanity/src/core/comments/index.ts create mode 100644 packages/sanity/src/core/comments/store/index.ts create mode 100644 packages/sanity/src/core/comments/store/reducer.ts create mode 100644 packages/sanity/src/core/comments/store/useCommentsStore.ts create mode 100644 packages/sanity/src/core/comments/types.ts create mode 100644 packages/sanity/src/desk/comments/field/CommentField.tsx create mode 100644 packages/sanity/src/desk/comments/field/CommentFieldButton.tsx create mode 100644 packages/sanity/src/desk/comments/field/index.ts create mode 100644 packages/sanity/src/desk/comments/index.ts create mode 100644 packages/sanity/src/desk/comments/inspector/CommentsInspector.tsx create mode 100644 packages/sanity/src/desk/comments/inspector/index.ts create mode 100644 packages/sanity/src/desk/comments/plugin.ts diff --git a/dev/test-studio/fieldActions/commentFieldAction.tsx b/dev/test-studio/fieldActions/commentFieldAction.tsx deleted file mode 100644 index 6bb6b356923d..000000000000 --- a/dev/test-studio/fieldActions/commentFieldAction.tsx +++ /dev/null @@ -1,22 +0,0 @@ -import {CommentIcon} from '@sanity/icons' -import {useCallback} from 'react' -import {defineDocumentFieldAction} from 'sanity' -import {defineActionItem} from './define' - -export const commentAction = defineDocumentFieldAction({ - name: 'test/comment', - useAction({documentId, documentType, path}) { - const onAction = useCallback(() => { - // eslint-disable-next-line no-console - console.log('comment', {documentId, documentType, path}) - }, [documentId, documentType, path]) - - return defineActionItem({ - type: 'action', - icon: CommentIcon, - onAction, - title: 'Comment', - renderAsButton: true, - }) - }, -}) diff --git a/dev/test-studio/sanity.config.ts b/dev/test-studio/sanity.config.ts index 12c708ed827e..139d59379ef6 100644 --- a/dev/test-studio/sanity.config.ts +++ b/dev/test-studio/sanity.config.ts @@ -39,7 +39,6 @@ import {vercelTheme} from './themes/vercel' import {GoogleLogo, TailwindLogo, VercelLogo} from './components/workspaceLogos' import {copyAction} from './fieldActions/copyAction' import {assistFieldActionGroup} from './fieldActions/assistFieldActionGroup' -import {commentAction} from './fieldActions/commentFieldAction' import {customInspector} from './inspectors/custom' import {pasteAction} from './fieldActions/pasteAction' @@ -70,7 +69,7 @@ const sharedSettings = definePlugin({ }, unstable_fieldActions: (prev, ctx) => { if (['fieldActionsTest', 'stringsTest'].includes(ctx.documentType)) { - return [...prev, commentAction, assistFieldActionGroup, copyAction, pasteAction] + return [...prev, assistFieldActionGroup, copyAction, pasteAction] } return prev diff --git a/packages/sanity/package.json b/packages/sanity/package.json index 17ff22e0d49b..2fd50fd0d038 100644 --- a/packages/sanity/package.json +++ b/packages/sanity/package.json @@ -151,6 +151,7 @@ "@sanity/logos": "^2.0.2", "@sanity/mutator": "3.16.2", "@sanity/portable-text-editor": "3.16.2", + "@portabletext/react": "^3.0.0", "@sanity/schema": "3.16.2", "@sanity/types": "3.16.2", "@sanity/ui": "^1.7.2", diff --git a/packages/sanity/src/core/comments/__workshop__/CommentInputStory.tsx b/packages/sanity/src/core/comments/__workshop__/CommentInputStory.tsx new file mode 100644 index 000000000000..7fef52209cfc --- /dev/null +++ b/packages/sanity/src/core/comments/__workshop__/CommentInputStory.tsx @@ -0,0 +1,42 @@ +import React, {useState} from 'react' +import {Card, Container, Flex} from '@sanity/ui' +import {PortableTextBlock} from '@sanity/types' +import {useBoolean} from '@sanity/ui-workshop' +import {CommentInput} from '../components' +import {CommentMessageSerializer} from '../components/pte' +import {useCurrentUser} from '../../store' + +export default function CommentsInputStory() { + const [value, setValue] = useState(null) + const currentUser = useCurrentUser() + const expandOnFocus = useBoolean('Expand on focus', false, 'Props') + const expanded = useBoolean('Expanded', false, 'Props') + + if (!currentUser) return null + + return ( + + + + + + + + + + + + + + + + ) +} diff --git a/packages/sanity/src/core/comments/__workshop__/CommentsListStory.tsx b/packages/sanity/src/core/comments/__workshop__/CommentsListStory.tsx new file mode 100644 index 000000000000..edf81eee17a2 --- /dev/null +++ b/packages/sanity/src/core/comments/__workshop__/CommentsListStory.tsx @@ -0,0 +1,112 @@ +import React, {useCallback, useState} from 'react' +import {Container, Flex} from '@sanity/ui' +import {CommentsList} from '../components' +import {useCurrentUser} from '../../store' +import {CommentDocument, CommentMessage, CommentCreatePayload} from '../types' + +const BASE: CommentDocument = { + _id: '1', + _type: 'comment', + _createdAt: new Date().toISOString(), + _updatedAt: '2021-05-04T14:54:37Z', + authorId: 'p8U8TipFc', + status: 'open', + _rev: '1', + path: {field: JSON.stringify([])}, + workspace: 'test', + target: { + documentId: '1', + documentType: 'article', + path: { + field: JSON.stringify([]), + }, + }, + message: [ + { + _type: 'block', + _key: '36a3f0d3832d', + style: 'normal', + markDefs: [], + children: [ + { + _type: 'span', + _key: '89014dd684ce', + text: 'My first comment', + marks: [], + }, + ], + }, + ], +} + +const PROPS = [ + { + ...BASE, + }, +] + +export default function CommentsListStory() { + const [state, setState] = useState(PROPS) + + const currentUser = useCurrentUser() + + const handleReplySubmit = useCallback( + (payload: CommentCreatePayload) => { + const comment: CommentDocument = { + ...BASE, + ...payload, + _createdAt: new Date().toISOString(), + _id: `${state.length + 1}`, + authorId: currentUser?.id || 'pP5s3g90N', + parentCommentId: payload.parentCommentId, + } + + setState((prev) => [...prev, comment]) + }, + [currentUser?.id, state.length], + ) + + const handleEdit = useCallback( + (id: string, payload: CommentCreatePayload) => { + setState((prev) => + prev.map((item) => { + if (item._id === id) { + return { + ...item, + ...payload, + _updatedAt: new Date().toISOString(), + } + } + + return item + }), + ) + }, + [setState], + ) + + const handleDelete = useCallback( + (id: string) => { + setState((prev) => prev.filter((item) => item._id !== id)) + }, + [setState], + ) + + if (!currentUser) return null + + return ( + + + + + + ) +} diff --git a/packages/sanity/src/core/comments/__workshop__/MentionOptionsHookStory.tsx b/packages/sanity/src/core/comments/__workshop__/MentionOptionsHookStory.tsx new file mode 100644 index 000000000000..937cc2b1ad97 --- /dev/null +++ b/packages/sanity/src/core/comments/__workshop__/MentionOptionsHookStory.tsx @@ -0,0 +1,21 @@ +import React from 'react' +import {Card, Code} from '@sanity/ui' +import {useMentionOptions} from '../hooks' + +export default function MentionOptionsHookStory() { + const {data, loading} = useMentionOptions({ + documentId: '1e1744ab-43d5-4fff-8a2a-28c58bf0434a', + documentType: 'author', + }) + + if (loading) return
Loading...
+ if (!data) return
No data
+ + return ( + + + {JSON.stringify(data, null, 2)} + + + ) +} diff --git a/packages/sanity/src/core/comments/__workshop__/MentionsMenuStory.tsx b/packages/sanity/src/core/comments/__workshop__/MentionsMenuStory.tsx new file mode 100644 index 000000000000..a7dbc3cf2c39 --- /dev/null +++ b/packages/sanity/src/core/comments/__workshop__/MentionsMenuStory.tsx @@ -0,0 +1,22 @@ +import {Container, Flex} from '@sanity/ui' +import React from 'react' +import {MentionsMenu} from '../components/mentions' +import {useMentionOptions} from '../hooks' + +export default function MentionsMenuStory() { + const {data, loading} = useMentionOptions({documentId: 'foo', documentType: 'bar'}) + + return ( + + + { + //... + }} + /> + + + ) +} diff --git a/packages/sanity/src/core/comments/__workshop__/index.ts b/packages/sanity/src/core/comments/__workshop__/index.ts new file mode 100644 index 000000000000..d3340ae229e4 --- /dev/null +++ b/packages/sanity/src/core/comments/__workshop__/index.ts @@ -0,0 +1,29 @@ +import {defineScope} from '@sanity/ui-workshop' +import {lazy} from 'react' + +export default defineScope({ + name: 'core/comments', + title: 'comments', + stories: [ + { + name: 'comments-input', + title: 'CommentsInput', + component: lazy(() => import('./CommentInputStory')), + }, + { + name: 'mention-options-hook', + title: 'useMentionOptions', + component: lazy(() => import('./MentionOptionsHookStory')), + }, + { + name: 'comments-list', + title: 'CommentsList', + component: lazy(() => import('./CommentsListStory')), + }, + { + name: 'mentions-menu', + title: 'MentionsMenu', + component: lazy(() => import('./MentionsMenuStory')), + }, + ], +}) diff --git a/packages/sanity/src/core/comments/components/CommentDeleteDialog.tsx b/packages/sanity/src/core/comments/components/CommentDeleteDialog.tsx new file mode 100644 index 000000000000..ee2d7151723b --- /dev/null +++ b/packages/sanity/src/core/comments/components/CommentDeleteDialog.tsx @@ -0,0 +1,70 @@ +import {Dialog, Grid, Button, Stack, Text} from '@sanity/ui' +import React, {useCallback} from 'react' + +const DIALOG_COPY: Record< + 'thread' | 'comment', + {title: string; body: string; confirmButtonText: string} +> = { + thread: { + title: 'Delete this comment thread?', + body: 'All comments in this thread will be deleted, and once deleted cannot be recovered.', + confirmButtonText: 'Delete thread', + }, + comment: { + title: 'Delete this comment?', + body: 'Once deleted, a comment cannot be recovered.', + confirmButtonText: 'Delete comment', + }, +} + +/** + * @beta + * @hidden + */ +export interface CommentDeleteDialogProps { + commentId: string + error: Error | null + isParent: boolean + loading: boolean + onClose: () => void + onDelete: (id: string) => void +} + +/** + * @beta + * @hidden + */ +export function CommentDeleteDialog(props: CommentDeleteDialogProps) { + const {isParent, onClose, commentId, onDelete, loading, error} = props + const {title, body, confirmButtonText} = DIALOG_COPY[isParent ? 'thread' : 'comment'] + + const handleDelete = useCallback(() => { + onDelete(commentId) + }, [commentId, onDelete]) + + return ( + + + ) +} diff --git a/packages/sanity/src/core/comments/components/constants.ts b/packages/sanity/src/core/comments/components/constants.ts new file mode 100644 index 000000000000..2377a8c1d062 --- /dev/null +++ b/packages/sanity/src/core/comments/components/constants.ts @@ -0,0 +1,4 @@ +import {FlexProps} from '@sanity/ui' + +export const FLEX_GAP: FlexProps['gap'] = 3 +export const AVATAR_SIZE = 25 diff --git a/packages/sanity/src/core/comments/components/icons/MentionIcon.tsx b/packages/sanity/src/core/comments/components/icons/MentionIcon.tsx new file mode 100644 index 000000000000..60c1e0a61735 --- /dev/null +++ b/packages/sanity/src/core/comments/components/icons/MentionIcon.tsx @@ -0,0 +1,27 @@ +import React, {forwardRef} from 'react' + +export const MentionIcon = forwardRef(function Icon( + props: React.SVGProps, + ref: React.Ref, +) { + return ( + + + + + ) +}) diff --git a/packages/sanity/src/core/comments/components/icons/SendIcon.tsx b/packages/sanity/src/core/comments/components/icons/SendIcon.tsx new file mode 100644 index 000000000000..ad0535013e56 --- /dev/null +++ b/packages/sanity/src/core/comments/components/icons/SendIcon.tsx @@ -0,0 +1,27 @@ +import React, {forwardRef} from 'react' + +export const SendIcon = forwardRef(function Icon( + props: React.SVGProps, + ref: React.Ref, +) { + return ( + + + + ) +}) diff --git a/packages/sanity/src/core/comments/components/icons/index.ts b/packages/sanity/src/core/comments/components/icons/index.ts new file mode 100644 index 000000000000..cbee6f9bee0a --- /dev/null +++ b/packages/sanity/src/core/comments/components/icons/index.ts @@ -0,0 +1,2 @@ +export * from './SendIcon' +export * from './MentionIcon' diff --git a/packages/sanity/src/core/comments/components/index.ts b/packages/sanity/src/core/comments/components/index.ts new file mode 100644 index 000000000000..1ccc700849fe --- /dev/null +++ b/packages/sanity/src/core/comments/components/index.ts @@ -0,0 +1,3 @@ +export * from './pte' +export * from './list' +export * from './CommentDeleteDialog' diff --git a/packages/sanity/src/core/comments/components/list/CommentsList.tsx b/packages/sanity/src/core/comments/components/list/CommentsList.tsx new file mode 100644 index 000000000000..1f4ce4e48bfd --- /dev/null +++ b/packages/sanity/src/core/comments/components/list/CommentsList.tsx @@ -0,0 +1,85 @@ +import React, {useMemo} from 'react' +import {Code, Stack} from '@sanity/ui' +import {CurrentUser} from '@sanity/types' +import {CommentCreatePayload, CommentDocument, CommentEditPayload} from '../../types' +import {CommentsListItem} from './CommentsListItem' + +interface GroupedComments { + [field: string]: CommentDocument[] +} + +function groupComments(comments: CommentDocument[]) { + return comments.reduce((acc, comment) => { + const field = comment.target?.path?.field + + if (!acc[field]) { + acc[field] = [] + } + + acc[field].push(comment) + + return acc + }, {} as GroupedComments) +} + +/** + * @beta + * @hidden + */ +export interface CommentsListProps { + comments: CommentDocument[] + currentUser: CurrentUser + documentId: string + documentType: string + onDelete: (id: string) => void + onEdit: (id: string, payload: CommentEditPayload) => void + onReply: (payload: CommentCreatePayload) => void +} + +/** + * @beta + * @hidden + */ +export function CommentsList(props: CommentsListProps) { + const {documentId, documentType, currentUser, comments, onReply, onDelete, onEdit} = props + const groupedComments = useMemo(() => groupComments(comments), [comments]) + + return ( + <> + + {Object.entries(groupedComments).map(([key, value]) => { + const currentComments = value + const parentComments = currentComments.filter((c) => !c.parentCommentId) + + return ( + + + + {key} + + + + {parentComments.map((comment) => { + const replies = currentComments.filter((c) => c.parentCommentId === comment._id) + + return ( + + ) + })} + + ) + })} + + + ) +} diff --git a/packages/sanity/src/core/comments/components/list/CommentsListItem.tsx b/packages/sanity/src/core/comments/components/list/CommentsListItem.tsx new file mode 100644 index 000000000000..b3c466422e1a --- /dev/null +++ b/packages/sanity/src/core/comments/components/list/CommentsListItem.tsx @@ -0,0 +1,98 @@ +import React, {useCallback, useState} from 'react' +import {Card, Stack} from '@sanity/ui' +import styled from 'styled-components' +import {CurrentUser} from '@sanity/types' +import {CommentInput} from '../pte/comment-input' +import { + CommentCreatePayload, + CommentDocument, + CommentEditPayload, + CommentMessage, +} from '../../types' +import {createNewComment} from '../../helpers' +import {CommentsListItemLayout} from './CommentsListItemLayout' + +const EMPTY_ARRAY: [] = [] + +const RootCard = styled(Card)`` + +interface CommentsListItemProps { + currentUser: CurrentUser + documentId: string + documentType: string + onDelete: (id: string) => void + onReply: (payload: CommentCreatePayload) => void + onEdit: (id: string, payload: CommentEditPayload) => void + parentComment: CommentDocument + replies: CommentDocument[] | undefined +} + +export function CommentsListItem(props: CommentsListItemProps) { + const {documentId, documentType, currentUser, parentComment, replies, onReply, onDelete, onEdit} = + props + const [value, setValue] = useState(EMPTY_ARRAY) + const hasReplies = replies && replies?.length > 0 + + const handleSubmit = useCallback(() => { + const nextComment = createNewComment({ + message: value, + parentCommentId: parentComment._id, + status: parentComment?.status || 'open', + target: parentComment.target, + }) + + if (nextComment) { + onReply?.(nextComment) + } + + setValue(EMPTY_ARRAY) + }, [onReply, parentComment._id, parentComment?.status, parentComment.target, value]) + + return ( + + + + + + {hasReplies && ( + <> + {replies.map((reply) => ( + + ))} + + )} + + + + + + ) +} diff --git a/packages/sanity/src/core/comments/components/list/CommentsListItemLayout.tsx b/packages/sanity/src/core/comments/components/list/CommentsListItemLayout.tsx new file mode 100644 index 000000000000..f12f679f107a --- /dev/null +++ b/packages/sanity/src/core/comments/components/list/CommentsListItemLayout.tsx @@ -0,0 +1,230 @@ +import {EditIcon, TrashIcon, EllipsisVerticalIcon} from '@sanity/icons' +import { + Avatar, + TextSkeleton, + Flex, + Button, + Stack, + Text, + Card, + useGlobalKeyDown, + Layer, + useClickOutside, + MenuButton, + Menu, + MenuItem, +} from '@sanity/ui' +import React, {useCallback, useState} from 'react' +import {CurrentUser} from '@sanity/types' +import styled, {css} from 'styled-components' +import {format} from 'date-fns' +import {UserAvatar} from '../../../components' +import {TimeAgoOpts, useTimeAgo} from '../../../hooks' +import {useUser} from '../../../store' +import {CommentMessageSerializer} from '../pte' +import {CommentInput} from '../pte/comment-input' +import {CommentDocument, CommentEditPayload, CommentMessage} from '../../types' +import {AVATAR_SIZE, FLEX_GAP} from '../constants' +import {useDidUpdate} from '../../../form' + +const FloatingLayer = styled(Layer)` + position: absolute; + top: 0; + right: 0; +` + +const FloatingCard = styled(Card)(({theme}) => { + const {space} = theme.sanity + + return css` + gap: ${space[1]}px; + padding: ${space[1] / 2}px; + ` +}) + +const RootStack = styled(Stack)` + position: relative; + + ${FloatingLayer}:not(:focus-within) { + opacity: 0; + } + + &[data-menu-open='true'] { + ${FloatingLayer} { + opacity: 1; + } + } + + @media (hover: hover) { + &:hover { + ${FloatingLayer} { + opacity: 1; + } + } + } +` + +interface CommentsListItemLayoutProps { + canDelete?: boolean + canEdit?: boolean + comment: CommentDocument + onDelete: (id: string) => void + onEdit: (id: string, message: CommentEditPayload) => void + currentUser: CurrentUser + documentId: string + documentType: string +} + +const TIME_AGO_OPTS: TimeAgoOpts = {agoSuffix: true} + +export function CommentsListItemLayout(props: CommentsListItemLayoutProps) { + const {canDelete, canEdit, comment, onDelete, currentUser, documentId, documentType, onEdit} = + props + const {_createdAt, authorId, message, _id, lastEditedAt} = comment + const [user] = useUser(authorId) + + const [value, setValue] = useState(message) + const [isEditing, setIsEditing] = useState(false) + const [rootElement, setRootElement] = useState(null) + const [mentionMenuOpen, setMentionMenuOpen] = useState(false) + + const [menuOpen, setMenuOpen] = useState(false) + + const _date = lastEditedAt || _createdAt + const date = _date ? new Date(_date) : new Date() + const timeAgo = useTimeAgo(date, TIME_AGO_OPTS) + + const handleMenuOpen = useCallback(() => setMenuOpen(true), []) + const handleMenuClose = useCallback(() => setMenuOpen(false), []) + + const handleDeleteClick = useCallback(() => { + onDelete(_id) + }, [_id, onDelete]) + + const handleEditSubmit = useCallback(() => { + onEdit(_id, { + message: value, + }) + setIsEditing(false) + }, [_id, onEdit, value]) + + const toggleEdit = useCallback(() => { + setIsEditing((v) => !v) + }, []) + + useDidUpdate(isEditing, () => setMenuOpen(false)) + + useGlobalKeyDown((event) => { + if (event.key === 'Escape') { + setIsEditing(false) + } + }) + + useClickOutside(() => { + if (!mentionMenuOpen) { + setIsEditing(false) + setValue(message) + } + }, [rootElement]) + + const avatar = user ? : + + const name = user?.displayName ? ( + + {user.displayName} + + ) : ( + // fix + ) + + return ( + + + {avatar} + + + {name} + + + {timeAgo} {lastEditedAt && <>(edited)} + + + + + {isEditing && ( + +
+ + + + + + )} + + {!isEditing && ( + <> + + + + } + onOpen={handleMenuOpen} + onClose={handleMenuClose} + menu={ + + + + + + } + /> + + + + + + + ) + } + + return ( + + + {count} comment{count > 1 ? 's' : ''} + + + } + > + + + ) +} diff --git a/packages/sanity/src/desk/comments/field/index.ts b/packages/sanity/src/desk/comments/field/index.ts new file mode 100644 index 000000000000..b20130880009 --- /dev/null +++ b/packages/sanity/src/desk/comments/field/index.ts @@ -0,0 +1 @@ +export * from './CommentField' diff --git a/packages/sanity/src/desk/comments/index.ts b/packages/sanity/src/desk/comments/index.ts new file mode 100644 index 000000000000..a0ed13843f9e --- /dev/null +++ b/packages/sanity/src/desk/comments/index.ts @@ -0,0 +1 @@ +export * from './plugin' diff --git a/packages/sanity/src/desk/comments/inspector/CommentsInspector.tsx b/packages/sanity/src/desk/comments/inspector/CommentsInspector.tsx new file mode 100644 index 000000000000..7112469a636e --- /dev/null +++ b/packages/sanity/src/desk/comments/inspector/CommentsInspector.tsx @@ -0,0 +1,141 @@ +import {Container, Flex, Spinner, Stack, Text} from '@sanity/ui' +import React, {useCallback, useEffect, useMemo, useState} from 'react' +import {DocumentInspectorHeader} from '../../panes/document/documentInspector' +import { + CommentCreatePayload, + CommentDeleteDialog, + CommentEditPayload, + CommentsList, + createNewComment, + DocumentInspectorProps, + useComments, + useCurrentUser, +} from 'sanity' + +export function CommentsInspector(props: DocumentInspectorProps) { + const {onClose, documentId, documentType} = props + const {comments, createOperation, editOperation, deleteOperation} = useComments() + const currentUser = useCurrentUser() + + const [showDeleteDialog, setShowDeleteDialog] = useState(false) + const [commentIdToDelete, setCommentIdToDelete] = useState<{ + commentId: string + isParent: boolean + } | null>(null) + + const handleReply = useCallback( + (payload: CommentCreatePayload) => { + const nextComment = createNewComment({ + documentId, + documentType, + message: payload.message, + parentCommentId: payload.parentCommentId, + status: payload.status, + target: payload.target, + }) + + createOperation.commit(nextComment) + }, + [createOperation, documentId, documentType], + ) + + const handleEdit = useCallback( + (id: string, payload: CommentEditPayload) => { + editOperation.commit(id, payload) + }, + [editOperation], + ) + + const handleDelete = useCallback( + (id: string) => { + deleteOperation.commit(id) + }, + [deleteOperation], + ) + + const closeDeleteDialog = useCallback(() => { + setShowDeleteDialog(false) + setCommentIdToDelete(null) + }, []) + + const onDeleteStart = useCallback( + (id: string) => { + setShowDeleteDialog(true) + setCommentIdToDelete({ + commentId: id, + isParent: comments.data.filter((c) => c.parentCommentId === id).length > 0, + }) + }, + [comments], + ) + + const commentRemoved = useMemo( + () => comments.data.find((c) => c._id === commentIdToDelete?.commentId) === undefined, + [commentIdToDelete?.commentId, comments.data], + ) + + // Close the dialog when the comment is successfully deleted + // todo: improve this + useEffect(() => { + if (commentRemoved && !deleteOperation.loading && !deleteOperation.error) { + closeDeleteDialog() + } + }, [closeDeleteDialog, commentRemoved, deleteOperation.error, deleteOperation.loading]) + + return ( + + {commentIdToDelete && showDeleteDialog && ( + + )} + + + + {comments.loading && ( + + + + + Loading comments... + + + + )} + + {!comments.loading && comments.data.length === 0 && ( + + + + + No comments yet + + + + + )} + + {comments.data.length > 0 && !comments.loading && currentUser && ( + + + + )} + + ) +} diff --git a/packages/sanity/src/desk/comments/inspector/index.ts b/packages/sanity/src/desk/comments/inspector/index.ts new file mode 100644 index 000000000000..98e8b8582ebf --- /dev/null +++ b/packages/sanity/src/desk/comments/inspector/index.ts @@ -0,0 +1,29 @@ +import {CommentIcon} from '@sanity/icons' +import {COMMENTS_INSPECTOR_NAME} from '../../panes/document/constants' +import {CommentsInspector} from './CommentsInspector' +import { + DocumentInspectorMenuItem, + DocumentInspectorUseMenuItemProps, + defineDocumentInspector, +} from 'sanity' + +function useMenuItem(props: DocumentInspectorUseMenuItemProps): DocumentInspectorMenuItem { + // eslint-disable-next-line no-empty-pattern + const { + // documentId, + // schemaType + } = props + + return { + icon: CommentIcon, + showAsAction: true, + title: 'Comments', + // tone: hasComments ? 'primary' : undefined, + } +} + +export const commentsInspector = defineDocumentInspector({ + name: COMMENTS_INSPECTOR_NAME, + component: CommentsInspector, + useMenuItem, +}) diff --git a/packages/sanity/src/desk/comments/plugin.ts b/packages/sanity/src/desk/comments/plugin.ts new file mode 100644 index 000000000000..0723e6c89d06 --- /dev/null +++ b/packages/sanity/src/desk/comments/plugin.ts @@ -0,0 +1,15 @@ +import {commentsInspector} from './inspector' +import {CommentField} from './field' +import {definePlugin} from 'sanity' + +export const comments = definePlugin({ + name: 'sanity/desk/comments', + document: { + inspectors: [commentsInspector], + }, + form: { + components: { + field: CommentField, + }, + }, +}) diff --git a/packages/sanity/src/desk/deskTool.ts b/packages/sanity/src/desk/deskTool.ts index 8ae9a53e33b3..cdba99fe06d4 100644 --- a/packages/sanity/src/desk/deskTool.ts +++ b/packages/sanity/src/desk/deskTool.ts @@ -12,8 +12,9 @@ import {LiveEditBadge} from './documentBadges' import {getIntentState} from './getIntentState' import {router} from './router' import {DeskToolOptions} from './types' -import {validationInspector} from './panes/document/inspectors/validation' +import {comments} from './comments' import {changesInspector} from './panes/document/inspectors/changes' +import {validationInspector} from './panes/document/inspectors/validation' import {definePlugin} from 'sanity' const documentActions = [ @@ -94,6 +95,9 @@ export const deskTool = definePlugin((options) => ({ return Array.from(new Set([...prevInspectors, ...inspectors])) }, }, + + plugins: [comments()], + tools: [ { name: options?.name || 'desk', diff --git a/packages/sanity/src/desk/panes/document/DocumentPaneProvider.tsx b/packages/sanity/src/desk/panes/document/DocumentPaneProvider.tsx index afc3804d08bd..7c7c15c6b78c 100644 --- a/packages/sanity/src/desk/panes/document/DocumentPaneProvider.tsx +++ b/packages/sanity/src/desk/panes/document/DocumentPaneProvider.tsx @@ -21,16 +21,25 @@ import { } from './constants' import {DocumentInspectorMenuItemsResolver} from './DocumentInspectorMenuItemsResolver' import { + CommentsProvider, + DocumentFieldAction, + DocumentFieldActionNode, DocumentInspector, + DocumentInspectorMenuItem, DocumentPresence, - PatchEvent, - StateTree, - toMutationPatches, + EMPTY_ARRAY, + FieldActionsProvider, + FieldActionsResolver, + getDraftId, getExpandOperations, getPublishedId, + PatchEvent, setAtPath, + StateTree, + toMutationPatches, useConnectionState, useDocumentOperation, + useDocumentValuePermissions, useEditState, useFormState, useInitialValue, @@ -38,18 +47,10 @@ import { useSchema, useSource, useTemplates, + useTimelineSelector, + useTimelineStore, useUnique, useValidationStatus, - getDraftId, - useDocumentValuePermissions, - useTimelineStore, - useTimelineSelector, - DocumentFieldAction, - DocumentInspectorMenuItem, - FieldActionsResolver, - EMPTY_ARRAY, - DocumentFieldActionNode, - FieldActionsProvider, } from 'sanity' /** @@ -683,7 +684,9 @@ export const DocumentPaneProvider = memo((props: DocumentPaneProviderProps) => { )} - {children} + + {children} + ) diff --git a/packages/sanity/src/desk/panes/document/constants.ts b/packages/sanity/src/desk/panes/document/constants.ts index 87e4ee92f9b6..a7c77188a09b 100644 --- a/packages/sanity/src/desk/panes/document/constants.ts +++ b/packages/sanity/src/desk/panes/document/constants.ts @@ -16,3 +16,4 @@ export const DEFAULT_MENU_ITEM_GROUPS: PaneMenuItemGroup[] = [{id: 'inspectors'} // inspectors export const HISTORY_INSPECTOR_NAME = 'sanity/desk/history' export const VALIDATION_INSPECTOR_NAME = 'sanity/desk/validation' +export const COMMENTS_INSPECTOR_NAME = 'sanity/desk/comments'