2
0
Files
bot/apps/builder/components/shared/Graph/Nodes/StepNode/StepNode.tsx

241 lines
7.1 KiB
TypeScript
Raw Normal View History

2022-01-06 16:54:23 +01:00
import {
Flex,
HStack,
Popover,
PopoverTrigger,
2022-01-19 09:44:21 +01:00
useDisclosure,
2022-01-06 16:54:23 +01:00
} from '@chakra-ui/react'
import React, { useEffect, useRef, useState } from 'react'
import {
BubbleStep,
BubbleStepContent,
DraggableStep,
Step,
TextBubbleContent,
TextBubbleStep,
} from 'models'
import { useGraph } from 'contexts/GraphContext'
import { StepIcon } from 'components/editor/StepsSideBar/StepIcon'
import { isBubbleStep, isTextBubbleStep } from 'utils'
2022-01-22 18:24:57 +01:00
import { StepNodeContent } from './StepNodeContent/StepNodeContent'
2022-01-12 09:10:59 +01:00
import { useTypebot } from 'contexts/TypebotContext'
import { ContextMenu } from 'components/shared/ContextMenu'
2022-01-06 16:54:23 +01:00
import { SettingsPopoverContent } from './SettingsPopoverContent'
2022-01-12 09:10:59 +01:00
import { StepNodeContextMenu } from './StepNodeContextMenu'
import { SourceEndpoint } from '../../Endpoints/SourceEndpoint'
2022-01-15 17:30:20 +01:00
import { hasDefaultConnector } from 'services/typebots'
import { useRouter } from 'next/router'
2022-01-19 09:44:21 +01:00
import { SettingsModal } from './SettingsPopoverContent/SettingsModal'
import { StepSettings } from './SettingsPopoverContent/SettingsPopoverContent'
import { TextBubbleEditor } from './TextBubbleEditor'
import { TargetEndpoint } from '../../Endpoints'
import { MediaBubblePopoverContent } from './MediaBubblePopoverContent'
import { NodePosition, useDragDistance } from 'contexts/GraphDndContext'
import { setMultipleRefs } from 'services/utils'
2021-12-16 10:43:49 +01:00
export const StepNode = ({
step,
isConnectable,
indices,
2021-12-16 10:43:49 +01:00
onMouseDown,
}: {
2022-01-06 09:40:56 +01:00
step: Step
2021-12-16 10:43:49 +01:00
isConnectable: boolean
indices: { stepIndex: number; blockIndex: number }
onMouseDown?: (stepNodePosition: NodePosition, step: DraggableStep) => void
2021-12-16 10:43:49 +01:00
}) => {
const { query } = useRouter()
const {
setConnectingIds,
connectingIds,
openedStepId,
setOpenedStepId,
setFocusedBlockId,
previewingEdge,
} = useGraph()
const { updateStep } = useTypebot()
2021-12-16 10:43:49 +01:00
const [isConnecting, setIsConnecting] = useState(false)
const [isPopoverOpened, setIsPopoverOpened] = useState(
openedStepId === step.id
)
const [isEditing, setIsEditing] = useState<boolean>(
isTextBubbleStep(step) && step.content.plainText === ''
)
const stepRef = useRef<HTMLDivElement | null>(null)
const isPreviewing = isConnecting || previewingEdge?.to.stepId === step.id
const onDrag = (position: NodePosition) => {
if (step.type === 'start' || !onMouseDown) return
onMouseDown(position, step)
}
useDragDistance({
ref: stepRef,
onDrag,
isDisabled: !onMouseDown || step.type === 'start',
})
2022-01-19 09:44:21 +01:00
const {
isOpen: isModalOpen,
onOpen: onModalOpen,
onClose: onModalClose,
} = useDisclosure()
2021-12-16 10:43:49 +01:00
useEffect(() => {
if (query.stepId?.toString() === step.id) setOpenedStepId(step.id)
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [query])
2021-12-16 10:43:49 +01:00
useEffect(() => {
setIsConnecting(
connectingIds?.target?.blockId === step.blockId &&
connectingIds?.target?.stepId === step.id
)
}, [connectingIds, step.blockId, step.id])
const handleModalClose = () => {
2022-03-09 15:12:00 +01:00
updateStep(indices, { ...step })
onModalClose()
}
2021-12-16 10:43:49 +01:00
const handleMouseEnter = () => {
if (connectingIds)
2021-12-16 10:43:49 +01:00
setConnectingIds({
...connectingIds,
target: { blockId: step.blockId, stepId: step.id },
2021-12-16 10:43:49 +01:00
})
}
const handleMouseLeave = () => {
if (connectingIds?.target)
setConnectingIds({
...connectingIds,
target: { ...connectingIds.target, stepId: undefined },
})
}
const handleCloseEditor = (content: TextBubbleContent) => {
2022-03-09 15:12:00 +01:00
const updatedStep = { ...step, content } as Step
updateStep(indices, updatedStep)
setIsEditing(false)
}
const handleClick = (e: React.MouseEvent) => {
setFocusedBlockId(step.blockId)
e.stopPropagation()
if (isTextBubbleStep(step)) setIsEditing(true)
setOpenedStepId(step.id)
}
const handleExpandClick = () => {
setOpenedStepId(undefined)
onModalOpen()
}
2022-03-09 15:12:00 +01:00
const handleStepUpdate = (updates: Partial<Step>) =>
updateStep(indices, { ...step, ...updates })
const handleContentChange = (content: BubbleStepContent) =>
2022-03-09 15:12:00 +01:00
updateStep(indices, { ...step, content } as Step)
useEffect(() => {
setIsPopoverOpened(openedStepId === step.id)
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [openedStepId])
2022-03-09 15:12:00 +01:00
return isEditing && isTextBubbleStep(step) ? (
<TextBubbleEditor
2022-03-09 15:12:00 +01:00
initialValue={step.content.richText}
onClose={handleCloseEditor}
/>
) : (
<ContextMenu<HTMLDivElement>
renderMenu={() => <StepNodeContextMenu indices={indices} />}
>
{(ref, isOpened) => (
<Popover
placement="left"
isLazy
isOpen={isPopoverOpened}
closeOnBlur={false}
>
2022-01-06 16:54:23 +01:00
<PopoverTrigger>
<Flex
pos="relative"
ref={setMultipleRefs([ref, stepRef])}
2022-01-06 16:54:23 +01:00
onMouseEnter={handleMouseEnter}
onMouseLeave={handleMouseLeave}
onClick={handleClick}
data-testid={`step`}
2022-01-12 09:10:59 +01:00
w="full"
2022-01-06 16:54:23 +01:00
>
<HStack
flex="1"
userSelect="none"
p="3"
borderWidth={isOpened || isPreviewing ? '2px' : '1px'}
borderColor={isOpened || isPreviewing ? 'blue.400' : 'gray.200'}
margin={isOpened || isPreviewing ? '-1px' : 0}
2022-01-06 16:54:23 +01:00
rounded="lg"
cursor={'pointer'}
2022-03-31 09:49:23 +02:00
bgColor="gray.50"
2022-01-12 09:10:59 +01:00
align="flex-start"
2022-01-22 18:24:57 +01:00
w="full"
2022-03-31 09:49:23 +02:00
transition="border-color 0.2s"
2022-01-06 16:54:23 +01:00
>
<StepIcon
2022-03-09 15:12:00 +01:00
type={step.type}
mt="1"
2022-03-09 15:12:00 +01:00
data-testid={`${step.id}-icon`}
/>
2022-03-09 15:12:00 +01:00
<StepNodeContent step={step} indices={indices} />
2022-01-15 17:30:20 +01:00
<TargetEndpoint
pos="absolute"
left="-32px"
top="19px"
2022-03-09 15:12:00 +01:00
stepId={step.id}
2022-01-15 17:30:20 +01:00
/>
2022-03-09 15:12:00 +01:00
{isConnectable && hasDefaultConnector(step) && (
<SourceEndpoint
source={{
2022-03-09 15:12:00 +01:00
blockId: step.blockId,
stepId: step.id,
}}
pos="absolute"
2022-03-31 09:49:23 +02:00
right="-34px"
bottom="10px"
/>
)}
2022-01-06 16:54:23 +01:00
</HStack>
</Flex>
</PopoverTrigger>
2022-03-09 15:12:00 +01:00
{hasSettingsPopover(step) && (
<SettingsPopoverContent
2022-03-09 15:12:00 +01:00
step={step}
onExpandClick={handleExpandClick}
2022-03-09 15:12:00 +01:00
onStepChange={handleStepUpdate}
/>
2022-01-19 09:44:21 +01:00
)}
2022-03-09 15:12:00 +01:00
{isMediaBubbleStep(step) && (
<MediaBubblePopoverContent
2022-03-09 15:12:00 +01:00
step={step}
onContentChange={handleContentChange}
/>
)}
<SettingsModal isOpen={isModalOpen} onClose={handleModalClose}>
2022-03-09 15:12:00 +01:00
<StepSettings step={step} onStepChange={handleStepUpdate} />
2022-01-19 09:44:21 +01:00
</SettingsModal>
2022-01-06 16:54:23 +01:00
</Popover>
2021-12-16 10:43:49 +01:00
)}
</ContextMenu>
2021-12-16 10:43:49 +01:00
)
}
2022-01-20 16:14:47 +01:00
const hasSettingsPopover = (step: Step): step is Exclude<Step, BubbleStep> =>
!isBubbleStep(step)
const isMediaBubbleStep = (
2022-01-20 16:14:47 +01:00
step: Step
): step is Exclude<BubbleStep, TextBubbleStep> =>
isBubbleStep(step) && !isTextBubbleStep(step)