index.tsx 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  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. handleNodeDuplicateSelected,
  107. handleNodeCopySelected,
  108. handleNodeCut,
  109. handleNodeDeleteSelected,
  110. handleNodePaste,
  111. } = useNodesInteractions()
  112. const {
  113. handleEdgeEnter,
  114. handleEdgeLeave,
  115. handleEdgeDelete,
  116. handleEdgesChange,
  117. } = useEdgesInteractions()
  118. const { isValidConnection } = useWorkflow()
  119. useOnViewportChange({
  120. onEnd: () => {
  121. handleSyncWorkflowDraft()
  122. },
  123. })
  124. useKeyPress(['delete'], handleEdgeDelete)
  125. useKeyPress(['delete'], handleNodeDeleteSelected)
  126. useKeyPress(['ctrl.c', 'meta.c'], handleNodeCopySelected)
  127. useKeyPress(['ctrl.x', 'meta.x'], handleNodeCut)
  128. useKeyPress(['ctrl.v', 'meta.v'], handleNodePaste)
  129. useKeyPress(['ctrl.alt.d', 'meta.shift.d'], handleNodeDuplicateSelected)
  130. return (
  131. <div
  132. id='workflow-container'
  133. className={`
  134. relative w-full min-w-[960px] h-full bg-[#F0F2F7]
  135. ${workflowReadOnly && 'workflow-panel-animation'}
  136. ${nodeAnimation && 'workflow-node-animation'}
  137. `}
  138. >
  139. <Header />
  140. <Panel />
  141. <Operator />
  142. {
  143. showFeaturesPanel && <Features />
  144. }
  145. <HelpLine />
  146. <ReactFlow
  147. nodeTypes={nodeTypes}
  148. edgeTypes={edgeTypes}
  149. nodes={nodes}
  150. edges={edges}
  151. onNodeDragStart={handleNodeDragStart}
  152. onNodeDrag={handleNodeDrag}
  153. onNodeDragStop={handleNodeDragStop}
  154. onNodeMouseEnter={handleNodeEnter}
  155. onNodeMouseLeave={handleNodeLeave}
  156. onNodeClick={handleNodeClick}
  157. onConnect={handleNodeConnect}
  158. onConnectStart={handleNodeConnectStart}
  159. onConnectEnd={handleNodeConnectEnd}
  160. onEdgeMouseEnter={handleEdgeEnter}
  161. onEdgeMouseLeave={handleEdgeLeave}
  162. onEdgesChange={handleEdgesChange}
  163. connectionLineComponent={CustomConnectionLine}
  164. defaultViewport={viewport}
  165. multiSelectionKeyCode={null}
  166. deleteKeyCode={null}
  167. nodesDraggable={!nodesReadOnly}
  168. nodesConnectable={!nodesReadOnly}
  169. nodesFocusable={!nodesReadOnly}
  170. edgesFocusable={!nodesReadOnly}
  171. panOnDrag={!workflowReadOnly}
  172. zoomOnPinch={!workflowReadOnly}
  173. zoomOnScroll={!workflowReadOnly}
  174. zoomOnDoubleClick={!workflowReadOnly}
  175. isValidConnection={isValidConnection}
  176. >
  177. <Background
  178. gap={[14, 14]}
  179. size={2}
  180. color='#E4E5E7'
  181. />
  182. </ReactFlow>
  183. </div>
  184. )
  185. })
  186. Workflow.displayName = 'Workflow'
  187. const WorkflowWrap = memo(() => {
  188. const {
  189. data,
  190. isLoading,
  191. } = useWorkflowInit()
  192. const nodesData = useMemo(() => {
  193. if (data)
  194. return initialNodes(data.graph.nodes, data.graph.edges)
  195. return []
  196. }, [data])
  197. const edgesData = useMemo(() => {
  198. if (data)
  199. return initialEdges(data.graph.edges, data.graph.nodes)
  200. return []
  201. }, [data])
  202. if (!data || isLoading) {
  203. return (
  204. <div className='flex justify-center items-center relative w-full h-full bg-[#F0F2F7]'>
  205. <Loading />
  206. </div>
  207. )
  208. }
  209. const features = data.features || {}
  210. const initialFeatures: FeaturesData = {
  211. file: {
  212. image: {
  213. enabled: !!features.file_upload?.image.enabled,
  214. number_limits: features.file_upload?.image.number_limits || 3,
  215. transfer_methods: features.file_upload?.image.transfer_methods || ['local_file', 'remote_url'],
  216. },
  217. },
  218. opening: {
  219. enabled: !!features.opening_statement,
  220. opening_statement: features.opening_statement,
  221. suggested_questions: features.suggested_questions,
  222. },
  223. suggested: features.suggested_questions_after_answer || { enabled: false },
  224. speech2text: features.speech_to_text || { enabled: false },
  225. text2speech: features.text_to_speech || { enabled: false },
  226. citation: features.retriever_resource || { enabled: false },
  227. moderation: features.sensitive_word_avoidance || { enabled: false },
  228. }
  229. return (
  230. <ReactFlowProvider>
  231. <FeaturesProvider features={initialFeatures}>
  232. <Workflow
  233. nodes={nodesData}
  234. edges={edgesData}
  235. viewport={data?.graph.viewport}
  236. />
  237. </FeaturesProvider>
  238. </ReactFlowProvider>
  239. )
  240. })
  241. WorkflowWrap.displayName = 'WorkflowWrap'
  242. const WorkflowContainer = () => {
  243. return (
  244. <WorkflowContextProvider>
  245. <WorkflowWrap />
  246. </WorkflowContextProvider>
  247. )
  248. }
  249. export default memo(WorkflowContainer)