index.tsx 6.5 KB

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