index.tsx 9.7 KB

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