import { useCallback, useMemo } from 'react'; import { BlockNoteEditor } from '@blocknote/core'; import { useBlockNote } from '@blocknote/react'; import styled from '@emotion/styled'; import { isArray, isNonEmptyString } from '@sniptt/guards'; import { useRecoilCallback, useRecoilState } from 'recoil'; import { Key } from 'ts-key-enum'; import { useDebouncedCallback } from 'use-debounce'; import { v4 } from 'uuid'; import { useUpsertActivity } from '@/activities/hooks/useUpsertActivity'; import { activityBodyFamilyState } from '@/activities/states/activityBodyFamilyState'; import { activityTitleHasBeenSetFamilyState } from '@/activities/states/activityTitleHasBeenSetFamilyState'; import { canCreateActivityState } from '@/activities/states/canCreateActivityState'; import { Activity } from '@/activities/types/Activity'; import { ActivityEditorHotkeyScope } from '@/activities/types/ActivityEditorHotkeyScope'; import { useObjectMetadataItemOnly } from '@/object-metadata/hooks/useObjectMetadataItemOnly'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { useModifyRecordFromCache } from '@/object-record/cache/hooks/useModifyRecordFromCache'; import { recordStoreFamilyState } from '@/object-record/record-store/states/recordStoreFamilyState'; import { BlockEditor } from '@/ui/input/editor/components/BlockEditor'; import { RightDrawerHotkeyScope } from '@/ui/layout/right-drawer/types/RightDrawerHotkeyScope'; import { usePreviousHotkeyScope } from '@/ui/utilities/hotkey/hooks/usePreviousHotkeyScope'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; import { isNonTextWritingKey } from '@/ui/utilities/hotkey/utils/isNonTextWritingKey'; import { REACT_APP_SERVER_BASE_URL } from '~/config'; import { FileFolder, useUploadFileMutation } from '~/generated/graphql'; import { blockSpecs } from '../blocks/blockSpecs'; import { getSlashMenu } from '../blocks/slashMenu'; import { getFileType } from '../files/utils/getFileType'; import '@blocknote/react/style.css'; const StyledBlockNoteStyledContainer = styled.div` height: 100%; width: 100%; `; type ActivityBodyEditorProps = { activityId: string; fillTitleFromBody: boolean; }; export const ActivityBodyEditor = ({ activityId, fillTitleFromBody, }: ActivityBodyEditorProps) => { const [activityInStore] = useRecoilState(recordStoreFamilyState(activityId)); const activity = activityInStore as Activity | null; const [activityTitleHasBeenSet, setActivityTitleHasBeenSet] = useRecoilState( activityTitleHasBeenSetFamilyState({ activityId: activityId, }), ); const [activityBody, setActivityBody] = useRecoilState( activityBodyFamilyState({ activityId: activityId, }), ); const { objectMetadataItem: objectMetadataItemActivity } = useObjectMetadataItemOnly({ objectNameSingular: CoreObjectNameSingular.Activity, }); const modifyActivityFromCache = useModifyRecordFromCache({ objectMetadataItem: objectMetadataItemActivity, }); const { goBackToPreviousHotkeyScope, setHotkeyScopeAndMemorizePreviousScope, } = usePreviousHotkeyScope(); const { upsertActivity } = useUpsertActivity(); const persistBodyDebounced = useDebouncedCallback((newBody: string) => { if (activity) { upsertActivity({ activity, input: { body: newBody, }, }); } }, 300); const persistTitleAndBodyDebounced = useDebouncedCallback( (newTitle: string, newBody: string) => { if (activity) { upsertActivity({ activity, input: { title: newTitle, body: newBody, }, }); setActivityTitleHasBeenSet(true); } }, 200, ); const updateTitleAndBody = useCallback( (newStringifiedBody: string) => { const blockBody = JSON.parse(newStringifiedBody); const newTitleFromBody = blockBody[0]?.content?.[0]?.text; persistTitleAndBodyDebounced(newTitleFromBody, newStringifiedBody); }, [persistTitleAndBodyDebounced], ); const [canCreateActivity, setCanCreateActivity] = useRecoilState( canCreateActivityState, ); const slashMenuItems = getSlashMenu(); const [uploadFile] = useUploadFileMutation(); const handleUploadAttachment = async (file: File): Promise => { if (!file) { return ''; } const result = await uploadFile({ variables: { file, fileFolder: FileFolder.Attachment, }, }); if (!result?.data?.uploadFile) { throw new Error("Couldn't upload Image"); } const imageUrl = REACT_APP_SERVER_BASE_URL + '/files/' + result?.data?.uploadFile; return imageUrl; }; const handlePersistBody = useCallback( (activityBody: string) => { if (!canCreateActivity) { setCanCreateActivity(true); } if (!activityTitleHasBeenSet && fillTitleFromBody) { updateTitleAndBody(activityBody); } else { persistBodyDebounced(activityBody); } }, [ fillTitleFromBody, persistBodyDebounced, activityTitleHasBeenSet, updateTitleAndBody, setCanCreateActivity, canCreateActivity, ], ); const handleBodyChange = useRecoilCallback( ({ snapshot, set }) => (newStringifiedBody: string) => { set(recordStoreFamilyState(activityId), (oldActivity) => { return { ...oldActivity, id: activityId, body: newStringifiedBody, }; }); modifyActivityFromCache(activityId, { body: () => { return newStringifiedBody; }, }); const activityTitleHasBeenSet = snapshot .getLoadable( activityTitleHasBeenSetFamilyState({ activityId: activityId, }), ) .getValue(); const blockBody = JSON.parse(newStringifiedBody); const newTitleFromBody = blockBody[0]?.content?.[0]?.text as string; if (!activityTitleHasBeenSet && fillTitleFromBody) { set(recordStoreFamilyState(activityId), (oldActivity) => { return { ...oldActivity, id: activityId, title: newTitleFromBody, }; }); modifyActivityFromCache(activityId, { title: () => { return newTitleFromBody; }, }); } handlePersistBody(newStringifiedBody); }, [activityId, fillTitleFromBody, modifyActivityFromCache, handlePersistBody], ); const handleBodyChangeDebounced = useDebouncedCallback(handleBodyChange, 500); const handleEditorChange = (newEditor: BlockNoteEditor) => { const newStringifiedBody = JSON.stringify(newEditor.topLevelBlocks) ?? ''; setActivityBody(newStringifiedBody); handleBodyChangeDebounced(newStringifiedBody); }; const initialBody = useMemo(() => { if (isNonEmptyString(activityBody) && activityBody !== '{}') { return JSON.parse(activityBody); } else if ( activity && isNonEmptyString(activity.body) && activity?.body !== '{}' ) { return JSON.parse(activity.body); } else { return undefined; } }, [activity, activityBody]); const editor: BlockNoteEditor | null = useBlockNote({ initialContent: initialBody, domAttributes: { editor: { class: 'editor' } }, onEditorContentChange: handleEditorChange, slashMenuItems, blockSpecs: blockSpecs, uploadFile: handleUploadAttachment, onEditorReady: (editor: BlockNoteEditor) => { editor.domElement.addEventListener('paste', handleImagePaste); }, }); const handleImagePaste = async (event: ClipboardEvent) => { const clipboardItems = event.clipboardData?.items; if (clipboardItems) { for (let i = 0; i < clipboardItems.length; i++) { if (clipboardItems[i].kind === 'file') { const isImage = clipboardItems[i].type.match('^image/'); const pastedFile = clipboardItems[i].getAsFile(); if (!pastedFile) { return; } const attachmentUrl = await handleUploadAttachment(pastedFile); if (!attachmentUrl) { return; } if (isImage) { editor?.insertBlocks( [ { type: 'image', props: { url: attachmentUrl, }, }, ], editor?.getTextCursorPosition().block, 'after', ); } else { editor?.insertBlocks( [ { type: 'file', props: { url: attachmentUrl, fileType: getFileType(pastedFile.name), name: pastedFile.name, }, }, ], editor?.getTextCursorPosition().block, 'after', ); } } } } }; useScopedHotkeys( Key.Escape, () => { editor.domElement?.blur(); }, ActivityEditorHotkeyScope.ActivityBody, ); useScopedHotkeys( '*', (keyboardEvent) => { if (keyboardEvent.key === Key.Escape) { return; } const isWritingText = !isNonTextWritingKey(keyboardEvent.key) && !keyboardEvent.ctrlKey && !keyboardEvent.metaKey; if (!isWritingText) { return; } keyboardEvent.preventDefault(); keyboardEvent.stopPropagation(); keyboardEvent.stopImmediatePropagation(); const blockIdentifier = editor.getTextCursorPosition().block; const currentBlockContent = blockIdentifier?.content; if ( currentBlockContent && isArray(currentBlockContent) && currentBlockContent.length === 0 ) { // Empty block case editor.updateBlock(blockIdentifier, { content: keyboardEvent.key, }); return; } if ( currentBlockContent && isArray(currentBlockContent) && currentBlockContent[0] && currentBlockContent[0].type === 'text' ) { // Text block case editor.updateBlock(blockIdentifier, { content: currentBlockContent[0].text + keyboardEvent.key, }); return; } const newBlockId = v4(); const newBlock = { id: newBlockId, type: 'paragraph', content: keyboardEvent.key, }; editor.insertBlocks([newBlock], blockIdentifier, 'after'); editor.setTextCursorPosition(newBlockId, 'end'); editor.focus(); }, RightDrawerHotkeyScope.RightDrawer, ); const handleBlockEditorFocus = () => { setHotkeyScopeAndMemorizePreviousScope( ActivityEditorHotkeyScope.ActivityBody, ); }; const handlerBlockEditorBlur = () => { goBackToPreviousHotkeyScope(); }; return ( editor.focus()}> ); };