mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-26 00:18:44 +00:00 
			
		
		
		
	 1a1bfd4048
			
		
	
	
		1a1bfd4048
		
			
		
	
	
	
	
		
			
			The frontend of feat: Persist Variables for Enhanced Debugging Workflow (#20699). Co-authored-by: jZonG <jzongcode@gmail.com>
		
			
				
	
	
		
			126 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			126 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import React, { useCallback, useRef, useState } from 'react'
 | |
| import { useTranslation } from 'react-i18next'
 | |
| import { useAppContext } from '@/context/app-context'
 | |
| import {
 | |
|   RiEqualizer2Line,
 | |
|   RiMenuLine,
 | |
| } from '@remixicon/react'
 | |
| import {
 | |
|   PortalToFollowElem,
 | |
|   PortalToFollowElemContent,
 | |
|   PortalToFollowElemTrigger,
 | |
| } from '@/app/components/base/portal-to-follow-elem'
 | |
| import AppIcon from '../base/app-icon'
 | |
| import Divider from '../base/divider'
 | |
| import AppInfo from './app-info'
 | |
| import NavLink from './navLink'
 | |
| import { useStore as useAppStore } from '@/app/components/app/store'
 | |
| import type { NavIcon } from './navLink'
 | |
| import cn from '@/utils/classnames'
 | |
| 
 | |
| type Props = {
 | |
|   navigation: Array<{
 | |
|     name: string
 | |
|     href: string
 | |
|     icon: NavIcon
 | |
|     selectedIcon: NavIcon
 | |
|   }>
 | |
| }
 | |
| 
 | |
| const AppSidebarDropdown = ({ navigation }: Props) => {
 | |
|   const { t } = useTranslation()
 | |
|   const { isCurrentWorkspaceEditor } = useAppContext()
 | |
|   const appDetail = useAppStore(state => state.appDetail)
 | |
|   const [detailExpand, setDetailExpand] = useState(false)
 | |
| 
 | |
|   const [open, doSetOpen] = useState(false)
 | |
|   const openRef = useRef(open)
 | |
|   const setOpen = useCallback((v: boolean) => {
 | |
|     doSetOpen(v)
 | |
|     openRef.current = v
 | |
|   }, [doSetOpen])
 | |
|   const handleTrigger = useCallback(() => {
 | |
|     setOpen(!openRef.current)
 | |
|   }, [setOpen])
 | |
| 
 | |
|   if (!appDetail)
 | |
|     return null
 | |
| 
 | |
|   return (
 | |
|     <>
 | |
|       <div className='fixed left-2 top-2 z-20'>
 | |
|         <PortalToFollowElem
 | |
|           open={open}
 | |
|           onOpenChange={setOpen}
 | |
|           placement='bottom-start'
 | |
|           offset={{
 | |
|             mainAxis: -41,
 | |
|           }}
 | |
|         >
 | |
|           <PortalToFollowElemTrigger onClick={handleTrigger}>
 | |
|             <div className={cn('flex cursor-pointer items-center rounded-[10px] border-[0.5px] border-components-actionbar-border bg-components-actionbar-bg p-1 shadow-lg backdrop-blur-sm hover:bg-background-default-hover', open && 'bg-background-default-hover')}>
 | |
|               <AppIcon
 | |
|                 size='small'
 | |
|                 iconType={appDetail.icon_type}
 | |
|                 icon={appDetail.icon}
 | |
|                 background={appDetail.icon_background}
 | |
|                 imageUrl={appDetail.icon_url}
 | |
|               />
 | |
|               <RiMenuLine className='h-4 w-4 text-text-tertiary' />
 | |
|             </div>
 | |
|           </PortalToFollowElemTrigger>
 | |
|           <PortalToFollowElemContent className='z-[1000]'>
 | |
|             <div className={cn('w-[305px] rounded-xl border-[0.5px] border-components-panel-border bg-background-default-subtle shadow-lg')}>
 | |
|               <div className='p-2'>
 | |
|                 <div
 | |
|                   className={cn('flex flex-col gap-2 rounded-lg p-2 pb-2.5', isCurrentWorkspaceEditor && 'cursor-pointer hover:bg-state-base-hover')}
 | |
|                   onClick={() => {
 | |
|                     setDetailExpand(true)
 | |
|                     setOpen(false)
 | |
|                   }}
 | |
|                 >
 | |
|                   <div className='flex items-center justify-between self-stretch'>
 | |
|                     <AppIcon
 | |
|                       size='large'
 | |
|                       iconType={appDetail.icon_type}
 | |
|                       icon={appDetail.icon}
 | |
|                       background={appDetail.icon_background}
 | |
|                       imageUrl={appDetail.icon_url}
 | |
|                     />
 | |
|                     <div className='flex items-center justify-center rounded-md p-0.5'>
 | |
|                       <div className='flex h-5 w-5 items-center justify-center'>
 | |
|                         <RiEqualizer2Line className='h-4 w-4 text-text-tertiary' />
 | |
|                       </div>
 | |
|                     </div>
 | |
|                   </div>
 | |
|                   <div className='flex flex-col items-start gap-1'>
 | |
|                     <div className='flex w-full'>
 | |
|                       <div className='system-md-semibold truncate text-text-secondary'>{appDetail.name}</div>
 | |
|                     </div>
 | |
|                     <div className='system-2xs-medium-uppercase text-text-tertiary'>{appDetail.mode === 'advanced-chat' ? t('app.types.advanced') : appDetail.mode === 'agent-chat' ? t('app.types.agent') : appDetail.mode === 'chat' ? t('app.types.chatbot') : appDetail.mode === 'completion' ? t('app.types.completion') : t('app.types.workflow')}</div>
 | |
|                   </div>
 | |
|                 </div>
 | |
|               </div>
 | |
|               <div className='px-4'>
 | |
|                 <Divider bgStyle='gradient' />
 | |
|               </div>
 | |
|               <nav className='space-y-0.5 px-3 pb-6 pt-4'>
 | |
|                 {navigation.map((item, index) => {
 | |
|                   return (
 | |
|                     <NavLink key={index} mode='expand' iconMap={{ selected: item.selectedIcon, normal: item.icon }} name={item.name} href={item.href} />
 | |
|                   )
 | |
|                 })}
 | |
|               </nav>
 | |
|             </div>
 | |
|           </PortalToFollowElemContent>
 | |
|         </PortalToFollowElem>
 | |
|       </div>
 | |
|       <div className='z-20'>
 | |
|         <AppInfo expand onlyShowDetail openState={detailExpand} onDetailExpand={setDetailExpand} />
 | |
|       </div>
 | |
|     </>
 | |
|   )
 | |
| }
 | |
| 
 | |
| export default AppSidebarDropdown
 |