index.tsx 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. 'use client'
  2. import type { FC } from 'react'
  3. import {
  4. memo,
  5. useCallback,
  6. useEffect,
  7. useMemo,
  8. } from 'react'
  9. import { setAutoFreeze } from 'immer'
  10. import {
  11. useKeyPress,
  12. } from 'ahooks'
  13. import ReactFlow, {
  14. Background,
  15. ReactFlowProvider,
  16. useEdgesState,
  17. useNodesState,
  18. useOnViewportChange,
  19. } from 'reactflow'
  20. import type { Viewport } from 'reactflow'
  21. import 'reactflow/dist/style.css'
  22. import './style.css'
  23. import type {
  24. Edge,
  25. Node,
  26. } from './types'
  27. import { WorkflowContextProvider } from './context'
  28. import {
  29. useEdgesInteractions,
  30. useNodesInteractions,
  31. useNodesReadOnly,
  32. useNodesSyncDraft,
  33. useWorkflow,
  34. useWorkflowInit,
  35. useWorkflowReadOnly,
  36. } from './hooks'
  37. import Header from './header'
  38. import CustomNode from './nodes'
  39. import Operator from './operator'
  40. import CustomEdge from './custom-edge'
  41. import CustomConnectionLine from './custom-connection-line'
  42. import Panel from './panel'
  43. import Features from './features'
  44. import HelpLine from './help-line'
  45. import { useStore } from './store'
  46. import {
  47. initialEdges,
  48. initialNodes,
  49. } from './utils'
  50. import { WORKFLOW_DATA_UPDATE } from './constants'
  51. import Loading from '@/app/components/base/loading'
  52. import { FeaturesProvider } from '@/app/components/base/features'
  53. import type { Features as FeaturesData } from '@/app/components/base/features/types'
  54. import { useEventEmitterContextContext } from '@/context/event-emitter'
  55. const nodeTypes = {
  56. custom: CustomNode,
  57. }
  58. const edgeTypes = {
  59. custom: CustomEdge,
  60. }
  61. type WorkflowProps = {
  62. nodes: Node[]
  63. edges: Edge[]
  64. viewport?: Viewport
  65. }
  66. const Workflow: FC<WorkflowProps> = memo(({
  67. nodes: originalNodes,
  68. edges: originalEdges,
  69. viewport,
  70. }) => {
  71. const [nodes, setNodes] = useNodesState(originalNodes)
  72. const [edges, setEdges] = useEdgesState(originalEdges)
  73. const showFeaturesPanel = useStore(state => state.showFeaturesPanel)
  74. const nodeAnimation = useStore(s => s.nodeAnimation)
  75. const {
  76. handleSyncWorkflowDraft,
  77. syncWorkflowDraftWhenPageClose,
  78. } = useNodesSyncDraft()
  79. const { workflowReadOnly } = useWorkflowReadOnly()
  80. const { nodesReadOnly } = useNodesReadOnly()
  81. const { eventEmitter } = useEventEmitterContextContext()
  82. eventEmitter?.useSubscription((v: any) => {
  83. if (v.type === WORKFLOW_DATA_UPDATE) {
  84. setNodes(v.payload.nodes)
  85. setEdges(v.payload.edges)
  86. }
  87. })
  88. useEffect(() => {
  89. setAutoFreeze(false)
  90. return () => {
  91. setAutoFreeze(true)
  92. }
  93. }, [])
  94. useEffect(() => {
  95. return () => {
  96. handleSyncWorkflowDraft(true)
  97. }
  98. }, [])
  99. const handleSyncWorkflowDraftWhenPageClose = useCallback(() => {
  100. if (document.visibilityState === 'hidden')
  101. syncWorkflowDraftWhenPageClose()
  102. }, [syncWorkflowDraftWhenPageClose])
  103. useEffect(() => {
  104. document.addEventListener('visibilitychange', handleSyncWorkflowDraftWhenPageClose)
  105. return () => {
  106. document.removeEventListener('visibilitychange', handleSyncWorkflowDraftWhenPageClose)
  107. }
  108. }, [handleSyncWorkflowDraftWhenPageClose])
  109. const {
  110. handleNodeDragStart,
  111. handleNodeDrag,
  112. handleNodeDragStop,
  113. handleNodeEnter,
  114. handleNodeLeave,
  115. handleNodeClick,
  116. handleNodeConnect,
  117. handleNodeConnectStart,
  118. handleNodeConnectEnd,
  119. handleNodeDuplicateSelected,
  120. handleNodeCopySelected,
  121. handleNodeCut,
  122. handleNodeDeleteSelected,
  123. handleNodePaste,
  124. } = useNodesInteractions()
  125. const {
  126. handleEdgeEnter,
  127. handleEdgeLeave,
  128. handleEdgeDelete,
  129. handleEdgesChange,
  130. } = useEdgesInteractions()
  131. const {
  132. isValidConnection,
  133. } = useWorkflow()
  134. useOnViewportChange({
  135. onEnd: () => {
  136. handleSyncWorkflowDraft()
  137. },
  138. })
  139. useKeyPress(['delete', 'backspace'], handleNodeDeleteSelected)
  140. useKeyPress(['delete', 'backspace'], handleEdgeDelete)
  141. useKeyPress(['ctrl.c', 'meta.c'], handleNodeCopySelected)
  142. useKeyPress(['ctrl.x', 'meta.x'], handleNodeCut)
  143. useKeyPress(['ctrl.v', 'meta.v'], handleNodePaste)
  144. useKeyPress(['ctrl.alt.d', 'meta.shift.d'], handleNodeDuplicateSelected)
  145. return (
  146. <div
  147. id='workflow-container'
  148. className={`
  149. relative w-full min-w-[960px] h-full bg-[#F0F2F7]
  150. ${workflowReadOnly && 'workflow-panel-animation'}
  151. ${nodeAnimation && 'workflow-node-animation'}
  152. `}
  153. >
  154. <Header />
  155. <Panel />
  156. <Operator />
  157. {
  158. showFeaturesPanel && <Features />
  159. }
  160. <HelpLine />
  161. <ReactFlow
  162. nodeTypes={nodeTypes}
  163. edgeTypes={edgeTypes}
  164. nodes={nodes}
  165. edges={edges}
  166. onNodeDragStart={handleNodeDragStart}
  167. onNodeDrag={handleNodeDrag}
  168. onNodeDragStop={handleNodeDragStop}
  169. onNodeMouseEnter={handleNodeEnter}
  170. onNodeMouseLeave={handleNodeLeave}
  171. onNodeClick={handleNodeClick}
  172. onConnect={handleNodeConnect}
  173. onConnectStart={handleNodeConnectStart}
  174. onConnectEnd={handleNodeConnectEnd}
  175. onEdgeMouseEnter={handleEdgeEnter}
  176. onEdgeMouseLeave={handleEdgeLeave}
  177. onEdgesChange={handleEdgesChange}
  178. connectionLineComponent={CustomConnectionLine}
  179. defaultViewport={viewport}
  180. multiSelectionKeyCode={null}
  181. deleteKeyCode={null}
  182. nodesDraggable={!nodesReadOnly}
  183. nodesConnectable={!nodesReadOnly}
  184. nodesFocusable={!nodesReadOnly}
  185. edgesFocusable={!nodesReadOnly}
  186. panOnDrag={!workflowReadOnly}
  187. zoomOnPinch={!workflowReadOnly}
  188. zoomOnScroll={!workflowReadOnly}
  189. zoomOnDoubleClick={!workflowReadOnly}
  190. isValidConnection={isValidConnection}
  191. >
  192. <Background
  193. gap={[14, 14]}
  194. size={2}
  195. color='#E4E5E7'
  196. />
  197. </ReactFlow>
  198. </div>
  199. )
  200. })
  201. Workflow.displayName = 'Workflow'
  202. const WorkflowWrap = memo(() => {
  203. const {
  204. data,
  205. isLoading,
  206. } = useWorkflowInit()
  207. const nodesData = useMemo(() => {
  208. if (data)
  209. return initialNodes(data.graph.nodes, data.graph.edges)
  210. return []
  211. }, [data])
  212. const edgesData = useMemo(() => {
  213. if (data)
  214. return initialEdges(data.graph.edges, data.graph.nodes)
  215. return []
  216. }, [data])
  217. if (!data || isLoading) {
  218. return (
  219. <div className='flex justify-center items-center relative w-full h-full bg-[#F0F2F7]'>
  220. <Loading />
  221. </div>
  222. )
  223. }
  224. const features = data.features || {}
  225. const initialFeatures: FeaturesData = {
  226. file: {
  227. image: {
  228. enabled: !!features.file_upload?.image.enabled,
  229. number_limits: features.file_upload?.image.number_limits || 3,
  230. transfer_methods: features.file_upload?.image.transfer_methods || ['local_file', 'remote_url'],
  231. },
  232. },
  233. opening: {
  234. enabled: !!features.opening_statement,
  235. opening_statement: features.opening_statement,
  236. suggested_questions: features.suggested_questions,
  237. },
  238. suggested: features.suggested_questions_after_answer || { enabled: false },
  239. speech2text: features.speech_to_text || { enabled: false },
  240. text2speech: features.text_to_speech || { enabled: false },
  241. citation: features.retriever_resource || { enabled: false },
  242. moderation: features.sensitive_word_avoidance || { enabled: false },
  243. }
  244. return (
  245. <ReactFlowProvider>
  246. <FeaturesProvider features={initialFeatures}>
  247. <Workflow
  248. nodes={nodesData}
  249. edges={edgesData}
  250. viewport={data?.graph.viewport}
  251. />
  252. </FeaturesProvider>
  253. </ReactFlowProvider>
  254. )
  255. })
  256. WorkflowWrap.displayName = 'WorkflowWrap'
  257. const WorkflowContainer = () => {
  258. return (
  259. <WorkflowContextProvider>
  260. <WorkflowWrap />
  261. </WorkflowContextProvider>
  262. )
  263. }
  264. export default memo(WorkflowContainer)