mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-24 23:48:40 +00:00 
			
		
		
		
	 7753ba2d37
			
		
	
	
		7753ba2d37
		
			
		
	
	
	
	
		
			
			Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: Yeuoly <admin@srmxy.cn> Co-authored-by: JzoNg <jzongcode@gmail.com> Co-authored-by: StyleZhang <jasonapring2015@outlook.com> Co-authored-by: jyong <jyong@dify.ai> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: jyong <718720800@qq.com>
		
			
				
	
	
		
			214 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			214 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import type { FC } from 'react'
 | |
| import {
 | |
|   memo,
 | |
|   useCallback,
 | |
| } from 'react'
 | |
| import { useTranslation } from 'react-i18next'
 | |
| import { useContext } from 'use-context-selector'
 | |
| import {
 | |
|   useStore,
 | |
|   useWorkflowStore,
 | |
| } from '../store'
 | |
| import {
 | |
|   useChecklistBeforePublish,
 | |
|   useNodesReadOnly,
 | |
|   useNodesSyncDraft,
 | |
|   useWorkflowRun,
 | |
| } from '../hooks'
 | |
| import AppPublisher from '../../app/app-publisher'
 | |
| import { ToastContext } from '../../base/toast'
 | |
| import RunAndHistory from './run-and-history'
 | |
| import EditingTitle from './editing-title'
 | |
| import RunningTitle from './running-title'
 | |
| import RestoringTitle from './restoring-title'
 | |
| import Checklist from './checklist'
 | |
| import { Grid01 } from '@/app/components/base/icons/src/vender/line/layout'
 | |
| import Button from '@/app/components/base/button'
 | |
| import { ArrowNarrowLeft } from '@/app/components/base/icons/src/vender/line/arrows'
 | |
| import { useStore as useAppStore } from '@/app/components/app/store'
 | |
| import { publishWorkflow } from '@/service/workflow'
 | |
| 
 | |
| const Header: FC = () => {
 | |
|   const { t } = useTranslation()
 | |
|   const workflowStore = useWorkflowStore()
 | |
|   const appDetail = useAppStore(s => s.appDetail)
 | |
|   const appSidebarExpand = useAppStore(s => s.appSidebarExpand)
 | |
|   const appID = useAppStore(state => state.appDetail?.id)
 | |
|   const {
 | |
|     nodesReadOnly,
 | |
|     getNodesReadOnly,
 | |
|   } = useNodesReadOnly()
 | |
|   const isRestoring = useStore(s => s.isRestoring)
 | |
|   const publishedAt = useStore(s => s.publishedAt)
 | |
|   const draftUpdatedAt = useStore(s => s.draftUpdatedAt)
 | |
|   const {
 | |
|     handleLoadBackupDraft,
 | |
|     handleRunSetting,
 | |
|     handleBackupDraft,
 | |
|     handleRestoreFromPublishedWorkflow,
 | |
|   } = useWorkflowRun()
 | |
|   const { handleCheckBeforePublish } = useChecklistBeforePublish()
 | |
|   const { handleSyncWorkflowDraft } = useNodesSyncDraft()
 | |
|   const { notify } = useContext(ToastContext)
 | |
| 
 | |
|   const handleShowFeatures = useCallback(() => {
 | |
|     const {
 | |
|       isRestoring,
 | |
|       setShowFeaturesPanel,
 | |
|     } = workflowStore.getState()
 | |
|     if (getNodesReadOnly() && !isRestoring)
 | |
|       return
 | |
| 
 | |
|     setShowFeaturesPanel(true)
 | |
|   }, [workflowStore, getNodesReadOnly])
 | |
| 
 | |
|   const handleGoBackToEdit = useCallback(() => {
 | |
|     handleRunSetting(true)
 | |
|   }, [handleRunSetting])
 | |
| 
 | |
|   const handleCancelRestore = useCallback(() => {
 | |
|     handleLoadBackupDraft()
 | |
|     workflowStore.setState({ isRestoring: false })
 | |
|   }, [workflowStore, handleLoadBackupDraft])
 | |
| 
 | |
|   const handleRestore = useCallback(() => {
 | |
|     workflowStore.setState({ isRestoring: false })
 | |
|     handleSyncWorkflowDraft(true)
 | |
|   }, [handleSyncWorkflowDraft, workflowStore])
 | |
| 
 | |
|   const onPublish = useCallback(async () => {
 | |
|     if (handleCheckBeforePublish()) {
 | |
|       const res = await publishWorkflow(`/apps/${appID}/workflows/publish`)
 | |
| 
 | |
|       if (res) {
 | |
|         notify({ type: 'success', message: t('common.api.actionSuccess') })
 | |
|         workflowStore.getState().setPublishedAt(res.created_at)
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       throw new Error('Checklist failed')
 | |
|     }
 | |
|   }, [appID, handleCheckBeforePublish, notify, t, workflowStore])
 | |
| 
 | |
|   const onStartRestoring = useCallback(() => {
 | |
|     workflowStore.setState({ isRestoring: true })
 | |
|     handleBackupDraft()
 | |
|     handleRestoreFromPublishedWorkflow()
 | |
|   }, [handleBackupDraft, handleRestoreFromPublishedWorkflow, workflowStore])
 | |
| 
 | |
|   const onPublisherToggle = useCallback((state: boolean) => {
 | |
|     if (state)
 | |
|       handleSyncWorkflowDraft(true)
 | |
|   }, [handleSyncWorkflowDraft])
 | |
| 
 | |
|   return (
 | |
|     <div
 | |
|       className='absolute top-0 left-0 z-10 flex items-center justify-between w-full px-3 h-14'
 | |
|       style={{
 | |
|         background: 'linear-gradient(180deg, #F9FAFB 0%, rgba(249, 250, 251, 0.00) 100%)',
 | |
|       }}
 | |
|     >
 | |
|       <div>
 | |
|         {
 | |
|           appSidebarExpand === 'collapse' && (
 | |
|             <div className='text-xs font-medium text-gray-700'>{appDetail?.name}</div>
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           !nodesReadOnly && !isRestoring && <EditingTitle />
 | |
|         }
 | |
|         {
 | |
|           nodesReadOnly && !isRestoring && <RunningTitle />
 | |
|         }
 | |
|         {
 | |
|           isRestoring && <RestoringTitle />
 | |
|         }
 | |
|       </div>
 | |
|       {
 | |
|         !isRestoring && (
 | |
|           <div className='flex items-center'>
 | |
|             {
 | |
|               nodesReadOnly && (
 | |
|                 <Button
 | |
|                   className={`
 | |
|                     mr-2 px-3 py-0 h-8 bg-white text-[13px] font-medium text-primary-600
 | |
|                     border-[0.5px] border-gray-200 shadow-xs
 | |
|                   `}
 | |
|                   onClick={handleGoBackToEdit}
 | |
|                 >
 | |
|                   <ArrowNarrowLeft className='w-4 h-4 mr-1' />
 | |
|                   {t('workflow.common.goBackToEdit')}
 | |
|                 </Button>
 | |
|               )
 | |
|             }
 | |
|             <RunAndHistory />
 | |
|             <div className='mx-2 w-[1px] h-3.5 bg-gray-200'></div>
 | |
|             <Button
 | |
|               className={`
 | |
|                 mr-2 px-3 py-0 h-8 bg-white text-[13px] font-medium text-gray-700
 | |
|                 border-[0.5px] border-gray-200 shadow-xs
 | |
|                 ${nodesReadOnly && !isRestoring && 'opacity-50 !cursor-not-allowed'}
 | |
|               `}
 | |
|               onClick={handleShowFeatures}
 | |
|             >
 | |
|               <Grid01 className='w-4 h-4 mr-1 text-gray-500' />
 | |
|               {t('workflow.common.features')}
 | |
|             </Button>
 | |
|             <AppPublisher
 | |
|               {...{
 | |
|                 publishedAt,
 | |
|                 draftUpdatedAt,
 | |
|                 disabled: Boolean(getNodesReadOnly()),
 | |
|                 onPublish,
 | |
|                 onRestore: onStartRestoring,
 | |
|                 onToggle: onPublisherToggle,
 | |
|                 crossAxisOffset: 53,
 | |
|               }}
 | |
|             />
 | |
|             {
 | |
|               !nodesReadOnly && (
 | |
|                 <>
 | |
|                   <div className='mx-2 w-[1px] h-3.5 bg-gray-200'></div>
 | |
|                   <Checklist />
 | |
|                 </>
 | |
|               )
 | |
|             }
 | |
|           </div>
 | |
|         )
 | |
|       }
 | |
|       {
 | |
|         isRestoring && (
 | |
|           <div className='flex items-center'>
 | |
|             <Button
 | |
|               className={`
 | |
|                 px-3 py-0 h-8 bg-white text-[13px] font-medium text-gray-700
 | |
|                 border-[0.5px] border-gray-200 shadow-xs
 | |
|               `}
 | |
|               onClick={handleShowFeatures}
 | |
|             >
 | |
|               <Grid01 className='w-4 h-4 mr-1 text-gray-500' />
 | |
|               {t('workflow.common.features')}
 | |
|             </Button>
 | |
|             <div className='mx-2 w-[1px] h-3.5 bg-gray-200'></div>
 | |
|             <Button
 | |
|               className='mr-2 px-3 py-0 h-8 bg-white text-[13px] text-gray-700 font-medium border-[0.5px] border-gray-200 shadow-xs'
 | |
|               onClick={handleCancelRestore}
 | |
|             >
 | |
|               {t('common.operation.cancel')}
 | |
|             </Button>
 | |
|             <Button
 | |
|               className='px-3 py-0 h-8 text-[13px] font-medium shadow-xs'
 | |
|               onClick={handleRestore}
 | |
|               type='primary'
 | |
|             >
 | |
|               {t('workflow.common.restore')}
 | |
|             </Button>
 | |
|           </div>
 | |
|         )
 | |
|       }
 | |
|     </div>
 | |
|   )
 | |
| }
 | |
| 
 | |
| export default memo(Header)
 |