index.tsx 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. 'use client'
  2. import type { FC } from 'react'
  3. import {
  4. memo,
  5. useCallback,
  6. useEffect,
  7. useMemo,
  8. useRef,
  9. } from 'react'
  10. import { setAutoFreeze } from 'immer'
  11. import {
  12. useEventListener,
  13. useKeyPress,
  14. } from 'ahooks'
  15. import ReactFlow, {
  16. Background,
  17. ReactFlowProvider,
  18. SelectionMode,
  19. useEdgesState,
  20. useNodesState,
  21. useOnViewportChange,
  22. } from 'reactflow'
  23. import type {
  24. Viewport,
  25. } from 'reactflow'
  26. import 'reactflow/dist/style.css'
  27. import './style.css'
  28. import type {
  29. Edge,
  30. Node,
  31. } from './types'
  32. import { WorkflowContextProvider } from './context'
  33. import {
  34. useEdgesInteractions,
  35. useNodesInteractions,
  36. useNodesReadOnly,
  37. useNodesSyncDraft,
  38. usePanelInteractions,
  39. useSelectionInteractions,
  40. useWorkflow,
  41. useWorkflowInit,
  42. useWorkflowReadOnly,
  43. useWorkflowStartRun,
  44. useWorkflowUpdate,
  45. } from './hooks'
  46. import Header from './header'
  47. import CustomNode from './nodes'
  48. import Operator from './operator'
  49. import CustomEdge from './custom-edge'
  50. import CustomConnectionLine from './custom-connection-line'
  51. import Panel from './panel'
  52. import Features from './features'
  53. import HelpLine from './help-line'
  54. import CandidateNode from './candidate-node'
  55. import PanelContextmenu from './panel-contextmenu'
  56. import NodeContextmenu from './node-contextmenu'
  57. import {
  58. useStore,
  59. useWorkflowStore,
  60. } from './store'
  61. import {
  62. getKeyboardKeyCodeBySystem,
  63. initialEdges,
  64. initialNodes,
  65. } from './utils'
  66. import {
  67. ITERATION_CHILDREN_Z_INDEX,
  68. WORKFLOW_DATA_UPDATE,
  69. } from './constants'
  70. import Loading from '@/app/components/base/loading'
  71. import { FeaturesProvider } from '@/app/components/base/features'
  72. import type { Features as FeaturesData } from '@/app/components/base/features/types'
  73. import { useEventEmitterContextContext } from '@/context/event-emitter'
  74. import Confirm from '@/app/components/base/confirm/common'
  75. const nodeTypes = {
  76. custom: CustomNode,
  77. }
  78. const edgeTypes = {
  79. custom: CustomEdge,
  80. }
  81. type WorkflowProps = {
  82. nodes: Node[]
  83. edges: Edge[]
  84. viewport?: Viewport
  85. }
  86. const Workflow: FC<WorkflowProps> = memo(({
  87. nodes: originalNodes,
  88. edges: originalEdges,
  89. viewport,
  90. }) => {
  91. const workflowContainerRef = useRef<HTMLDivElement>(null)
  92. const workflowStore = useWorkflowStore()
  93. const [nodes, setNodes] = useNodesState(originalNodes)
  94. const [edges, setEdges] = useEdgesState(originalEdges)
  95. const showFeaturesPanel = useStore(state => state.showFeaturesPanel)
  96. const controlMode = useStore(s => s.controlMode)
  97. const nodeAnimation = useStore(s => s.nodeAnimation)
  98. const showConfirm = useStore(s => s.showConfirm)
  99. const { setShowConfirm } = workflowStore.getState()
  100. const {
  101. handleSyncWorkflowDraft,
  102. syncWorkflowDraftWhenPageClose,
  103. } = useNodesSyncDraft()
  104. const { workflowReadOnly } = useWorkflowReadOnly()
  105. const { nodesReadOnly } = useNodesReadOnly()
  106. const { eventEmitter } = useEventEmitterContextContext()
  107. eventEmitter?.useSubscription((v: any) => {
  108. if (v.type === WORKFLOW_DATA_UPDATE) {
  109. setNodes(v.payload.nodes)
  110. setEdges(v.payload.edges)
  111. }
  112. })
  113. useEffect(() => {
  114. setAutoFreeze(false)
  115. return () => {
  116. setAutoFreeze(true)
  117. }
  118. }, [])
  119. useEffect(() => {
  120. return () => {
  121. handleSyncWorkflowDraft(true, true)
  122. }
  123. }, [])
  124. const { handleRefreshWorkflowDraft } = useWorkflowUpdate()
  125. const handleSyncWorkflowDraftWhenPageClose = useCallback(() => {
  126. if (document.visibilityState === 'hidden')
  127. syncWorkflowDraftWhenPageClose()
  128. else if (document.visibilityState === 'visible')
  129. handleRefreshWorkflowDraft()
  130. }, [syncWorkflowDraftWhenPageClose, handleRefreshWorkflowDraft])
  131. useEffect(() => {
  132. document.addEventListener('visibilitychange', handleSyncWorkflowDraftWhenPageClose)
  133. return () => {
  134. document.removeEventListener('visibilitychange', handleSyncWorkflowDraftWhenPageClose)
  135. }
  136. }, [handleSyncWorkflowDraftWhenPageClose])
  137. useEventListener('keydown', (e) => {
  138. if ((e.key === 'd' || e.key === 'D') && (e.ctrlKey || e.metaKey))
  139. e.preventDefault()
  140. })
  141. useEventListener('mousemove', (e) => {
  142. const containerClientRect = workflowContainerRef.current?.getBoundingClientRect()
  143. if (containerClientRect) {
  144. workflowStore.setState({
  145. mousePosition: {
  146. pageX: e.clientX,
  147. pageY: e.clientY,
  148. elementX: e.clientX - containerClientRect.left,
  149. elementY: e.clientY - containerClientRect.top,
  150. },
  151. })
  152. }
  153. })
  154. const {
  155. handleNodeDragStart,
  156. handleNodeDrag,
  157. handleNodeDragStop,
  158. handleNodeEnter,
  159. handleNodeLeave,
  160. handleNodeClick,
  161. handleNodeConnect,
  162. handleNodeConnectStart,
  163. handleNodeConnectEnd,
  164. handleNodeContextMenu,
  165. handleNodesCopy,
  166. handleNodesPaste,
  167. handleNodesDuplicate,
  168. handleNodesDelete,
  169. } = useNodesInteractions()
  170. const {
  171. handleEdgeEnter,
  172. handleEdgeLeave,
  173. handleEdgeDelete,
  174. handleEdgesChange,
  175. } = useEdgesInteractions()
  176. const {
  177. handleSelectionStart,
  178. handleSelectionChange,
  179. handleSelectionDrag,
  180. } = useSelectionInteractions()
  181. const {
  182. handlePaneContextMenu,
  183. } = usePanelInteractions()
  184. const {
  185. isValidConnection,
  186. } = useWorkflow()
  187. const { handleStartWorkflowRun } = useWorkflowStartRun()
  188. useOnViewportChange({
  189. onEnd: () => {
  190. handleSyncWorkflowDraft()
  191. },
  192. })
  193. useKeyPress('delete', handleNodesDelete)
  194. useKeyPress(['delete', 'backspace'], handleEdgeDelete)
  195. useKeyPress(`${getKeyboardKeyCodeBySystem('ctrl')}.c`, handleNodesCopy, { exactMatch: true, useCapture: true })
  196. useKeyPress(`${getKeyboardKeyCodeBySystem('ctrl')}.v`, handleNodesPaste, { exactMatch: true, useCapture: true })
  197. useKeyPress(`${getKeyboardKeyCodeBySystem('ctrl')}.d`, handleNodesDuplicate, { exactMatch: true, useCapture: true })
  198. useKeyPress(`${getKeyboardKeyCodeBySystem('alt')}.r`, handleStartWorkflowRun, { exactMatch: true, useCapture: true })
  199. return (
  200. <div
  201. id='workflow-container'
  202. className={`
  203. relative w-full min-w-[960px] h-full bg-[#F0F2F7]
  204. ${workflowReadOnly && 'workflow-panel-animation'}
  205. ${nodeAnimation && 'workflow-node-animation'}
  206. `}
  207. ref={workflowContainerRef}
  208. >
  209. <CandidateNode />
  210. <Header />
  211. <Panel />
  212. <Operator />
  213. {
  214. showFeaturesPanel && <Features />
  215. }
  216. <PanelContextmenu />
  217. <NodeContextmenu />
  218. <HelpLine />
  219. {
  220. !!showConfirm && (
  221. <Confirm
  222. isShow
  223. onCancel={() => setShowConfirm(undefined)}
  224. onConfirm={showConfirm.onConfirm}
  225. title={showConfirm.title}
  226. desc={showConfirm.desc}
  227. confirmWrapperClassName='!z-[11]'
  228. />
  229. )
  230. }
  231. <ReactFlow
  232. nodeTypes={nodeTypes}
  233. edgeTypes={edgeTypes}
  234. nodes={nodes}
  235. edges={edges}
  236. onNodeDragStart={handleNodeDragStart}
  237. onNodeDrag={handleNodeDrag}
  238. onNodeDragStop={handleNodeDragStop}
  239. onNodeMouseEnter={handleNodeEnter}
  240. onNodeMouseLeave={handleNodeLeave}
  241. onNodeClick={handleNodeClick}
  242. onNodeContextMenu={handleNodeContextMenu}
  243. onConnect={handleNodeConnect}
  244. onConnectStart={handleNodeConnectStart}
  245. onConnectEnd={handleNodeConnectEnd}
  246. onEdgeMouseEnter={handleEdgeEnter}
  247. onEdgeMouseLeave={handleEdgeLeave}
  248. onEdgesChange={handleEdgesChange}
  249. onSelectionStart={handleSelectionStart}
  250. onSelectionChange={handleSelectionChange}
  251. onSelectionDrag={handleSelectionDrag}
  252. onPaneContextMenu={handlePaneContextMenu}
  253. connectionLineComponent={CustomConnectionLine}
  254. connectionLineContainerStyle={{ zIndex: ITERATION_CHILDREN_Z_INDEX }}
  255. defaultViewport={viewport}
  256. multiSelectionKeyCode={null}
  257. deleteKeyCode={null}
  258. nodesDraggable={!nodesReadOnly}
  259. nodesConnectable={!nodesReadOnly}
  260. nodesFocusable={!nodesReadOnly}
  261. edgesFocusable={!nodesReadOnly}
  262. panOnDrag={controlMode === 'hand' && !workflowReadOnly}
  263. zoomOnPinch={!workflowReadOnly}
  264. zoomOnScroll={!workflowReadOnly}
  265. zoomOnDoubleClick={!workflowReadOnly}
  266. isValidConnection={isValidConnection}
  267. selectionKeyCode={null}
  268. selectionMode={SelectionMode.Partial}
  269. selectionOnDrag={controlMode === 'pointer' && !workflowReadOnly}
  270. minZoom={0.25}
  271. >
  272. <Background
  273. gap={[14, 14]}
  274. size={2}
  275. color='#E4E5E7'
  276. />
  277. </ReactFlow>
  278. </div>
  279. )
  280. })
  281. Workflow.displayName = 'Workflow'
  282. const WorkflowWrap = memo(() => {
  283. const {
  284. data,
  285. isLoading,
  286. } = useWorkflowInit()
  287. const nodesData = useMemo(() => {
  288. if (data)
  289. return initialNodes(data.graph.nodes, data.graph.edges)
  290. return []
  291. }, [data])
  292. const edgesData = useMemo(() => {
  293. if (data)
  294. return initialEdges(data.graph.edges, data.graph.nodes)
  295. return []
  296. }, [data])
  297. if (!data || isLoading) {
  298. return (
  299. <div className='flex justify-center items-center relative w-full h-full bg-[#F0F2F7]'>
  300. <Loading />
  301. </div>
  302. )
  303. }
  304. const features = data.features || {}
  305. const initialFeatures: FeaturesData = {
  306. file: {
  307. image: {
  308. enabled: !!features.file_upload?.image.enabled,
  309. number_limits: features.file_upload?.image.number_limits || 3,
  310. transfer_methods: features.file_upload?.image.transfer_methods || ['local_file', 'remote_url'],
  311. },
  312. },
  313. opening: {
  314. enabled: !!features.opening_statement,
  315. opening_statement: features.opening_statement,
  316. suggested_questions: features.suggested_questions,
  317. },
  318. suggested: features.suggested_questions_after_answer || { enabled: false },
  319. speech2text: features.speech_to_text || { enabled: false },
  320. text2speech: features.text_to_speech || { enabled: false },
  321. citation: features.retriever_resource || { enabled: false },
  322. moderation: features.sensitive_word_avoidance || { enabled: false },
  323. }
  324. return (
  325. <ReactFlowProvider>
  326. <FeaturesProvider features={initialFeatures}>
  327. <Workflow
  328. nodes={nodesData}
  329. edges={edgesData}
  330. viewport={data?.graph.viewport}
  331. />
  332. </FeaturesProvider>
  333. </ReactFlowProvider>
  334. )
  335. })
  336. WorkflowWrap.displayName = 'WorkflowWrap'
  337. const WorkflowContainer = () => {
  338. return (
  339. <WorkflowContextProvider>
  340. <WorkflowWrap />
  341. </WorkflowContextProvider>
  342. )
  343. }
  344. export default memo(WorkflowContainer)