| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import type { FC } from 'react' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   Fragment, | 
					
						
							|  |  |  |   memo, | 
					
						
							|  |  |  |   useCallback, | 
					
						
							|  |  |  |   useState, | 
					
						
							|  |  |  | } from 'react' | 
					
						
							| 
									
										
										
										
											2024-06-20 11:05:08 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   RiZoomInLine, | 
					
						
							|  |  |  |   RiZoomOutLine, | 
					
						
							|  |  |  | } from '@remixicon/react' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   useReactFlow, | 
					
						
							|  |  |  |   useViewport, | 
					
						
							|  |  |  | } from 'reactflow' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   useNodesSyncDraft, | 
					
						
							|  |  |  |   useWorkflowReadOnly, | 
					
						
							|  |  |  | } from '../hooks' | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   getKeyboardKeyNameBySystem, | 
					
						
							|  |  |  | } from '../utils' | 
					
						
							|  |  |  | import ShortcutsName from '../shortcuts-name' | 
					
						
							|  |  |  | import TipPopup from './tip-popup' | 
					
						
							| 
									
										
										
										
											2024-07-09 15:05:40 +08:00
										 |  |  | import cn from '@/utils/classnames' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   PortalToFollowElem, | 
					
						
							|  |  |  |   PortalToFollowElemContent, | 
					
						
							|  |  |  |   PortalToFollowElemTrigger, | 
					
						
							|  |  |  | } from '@/app/components/base/portal-to-follow-elem' | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum ZoomType { | 
					
						
							|  |  |  |   zoomIn = 'zoomIn', | 
					
						
							|  |  |  |   zoomOut = 'zoomOut', | 
					
						
							|  |  |  |   zoomToFit = 'zoomToFit', | 
					
						
							|  |  |  |   zoomTo25 = 'zoomTo25', | 
					
						
							|  |  |  |   zoomTo50 = 'zoomTo50', | 
					
						
							|  |  |  |   zoomTo75 = 'zoomTo75', | 
					
						
							|  |  |  |   zoomTo100 = 'zoomTo100', | 
					
						
							|  |  |  |   zoomTo200 = 'zoomTo200', | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | const ZoomInOut: FC = () => { | 
					
						
							|  |  |  |   const { t } = useTranslation() | 
					
						
							|  |  |  |   const { | 
					
						
							|  |  |  |     zoomIn, | 
					
						
							|  |  |  |     zoomOut, | 
					
						
							|  |  |  |     zoomTo, | 
					
						
							|  |  |  |     fitView, | 
					
						
							|  |  |  |   } = useReactFlow() | 
					
						
							|  |  |  |   const { zoom } = useViewport() | 
					
						
							|  |  |  |   const { handleSyncWorkflowDraft } = useNodesSyncDraft() | 
					
						
							|  |  |  |   const [open, setOpen] = useState(false) | 
					
						
							|  |  |  |   const { | 
					
						
							|  |  |  |     workflowReadOnly, | 
					
						
							|  |  |  |     getWorkflowReadOnly, | 
					
						
							|  |  |  |   } = useWorkflowReadOnly() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const ZOOM_IN_OUT_OPTIONS = [ | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |         key: ZoomType.zoomTo200, | 
					
						
							|  |  |  |         text: '200%', | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       }, | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |         key: ZoomType.zoomTo100, | 
					
						
							|  |  |  |         text: '100%', | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       }, | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |         key: ZoomType.zoomTo75, | 
					
						
							|  |  |  |         text: '75%', | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       }, | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |         key: ZoomType.zoomTo50, | 
					
						
							|  |  |  |         text: '50%', | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         key: ZoomType.zoomTo25, | 
					
						
							|  |  |  |         text: '25%', | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       }, | 
					
						
							|  |  |  |     ], | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |         key: ZoomType.zoomToFit, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         text: t('workflow.operator.zoomToFit'), | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ], | 
					
						
							|  |  |  |   ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleZoom = (type: string) => { | 
					
						
							|  |  |  |     if (workflowReadOnly) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |     if (type === ZoomType.zoomToFit) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       fitView() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |     if (type === ZoomType.zoomTo25) | 
					
						
							|  |  |  |       zoomTo(0.25) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (type === ZoomType.zoomTo50) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       zoomTo(0.5) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |     if (type === ZoomType.zoomTo75) | 
					
						
							|  |  |  |       zoomTo(0.75) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (type === ZoomType.zoomTo100) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       zoomTo(1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |     if (type === ZoomType.zoomTo200) | 
					
						
							|  |  |  |       zoomTo(2) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     handleSyncWorkflowDraft() | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleTrigger = useCallback(() => { | 
					
						
							|  |  |  |     if (getWorkflowReadOnly()) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setOpen(v => !v) | 
					
						
							|  |  |  |   }, [getWorkflowReadOnly]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <PortalToFollowElem | 
					
						
							|  |  |  |       placement='top-start' | 
					
						
							|  |  |  |       open={open} | 
					
						
							|  |  |  |       onOpenChange={setOpen} | 
					
						
							|  |  |  |       offset={{ | 
					
						
							|  |  |  |         mainAxis: 4, | 
					
						
							|  |  |  |         crossAxis: -2, | 
					
						
							|  |  |  |       }} | 
					
						
							|  |  |  |     > | 
					
						
							|  |  |  |       <PortalToFollowElemTrigger asChild onClick={handleTrigger}> | 
					
						
							|  |  |  |         <div className={`
 | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |           p-0.5 h-9 cursor-pointer text-[13px] text-gray-500 font-medium rounded-lg bg-white shadow-lg border-[0.5px] border-gray-100 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |           ${workflowReadOnly && '!cursor-not-allowed opacity-50'} | 
					
						
							|  |  |  |         `}>
 | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |           <div className={cn( | 
					
						
							|  |  |  |             'flex items-center justify-between w-[98px] h-8 hover:bg-gray-50 rounded-lg', | 
					
						
							|  |  |  |             open && 'bg-gray-50', | 
					
						
							|  |  |  |           )}> | 
					
						
							|  |  |  |             <TipPopup | 
					
						
							|  |  |  |               title={t('workflow.operator.zoomOut')} | 
					
						
							|  |  |  |               shortcuts={['ctrl', '-']} | 
					
						
							|  |  |  |             > | 
					
						
							|  |  |  |               <div | 
					
						
							|  |  |  |                 className='flex items-center justify-center w-8 h-8 rounded-lg cursor-pointer hover:bg-black/5' | 
					
						
							|  |  |  |                 onClick={(e) => { | 
					
						
							|  |  |  |                   e.stopPropagation() | 
					
						
							|  |  |  |                   zoomOut() | 
					
						
							|  |  |  |                 }} | 
					
						
							|  |  |  |               > | 
					
						
							| 
									
										
										
										
											2024-06-20 11:05:08 +08:00
										 |  |  |                 <RiZoomOutLine className='w-4 h-4' /> | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |               </div> | 
					
						
							|  |  |  |             </TipPopup> | 
					
						
							|  |  |  |             <div className='w-[34px]'>{parseFloat(`${zoom * 100}`).toFixed(0)}%</div> | 
					
						
							|  |  |  |             <TipPopup | 
					
						
							|  |  |  |               title={t('workflow.operator.zoomIn')} | 
					
						
							|  |  |  |               shortcuts={['ctrl', '+']} | 
					
						
							|  |  |  |             > | 
					
						
							|  |  |  |               <div | 
					
						
							|  |  |  |                 className='flex items-center justify-center w-8 h-8 rounded-lg cursor-pointer hover:bg-black/5' | 
					
						
							|  |  |  |                 onClick={(e) => { | 
					
						
							|  |  |  |                   e.stopPropagation() | 
					
						
							|  |  |  |                   zoomIn() | 
					
						
							|  |  |  |                 }} | 
					
						
							|  |  |  |               > | 
					
						
							| 
									
										
										
										
											2024-06-20 11:05:08 +08:00
										 |  |  |                 <RiZoomInLine className='w-4 h-4' /> | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |               </div> | 
					
						
							|  |  |  |             </TipPopup> | 
					
						
							|  |  |  |           </div> | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         </div> | 
					
						
							|  |  |  |       </PortalToFollowElemTrigger> | 
					
						
							|  |  |  |       <PortalToFollowElemContent className='z-10'> | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |         <div className='w-[145px] rounded-lg border-[0.5px] border-gray-200 bg-white shadow-lg'> | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |           { | 
					
						
							|  |  |  |             ZOOM_IN_OUT_OPTIONS.map((options, i) => ( | 
					
						
							|  |  |  |               <Fragment key={i}> | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                   i !== 0 && ( | 
					
						
							|  |  |  |                     <div className='h-[1px] bg-gray-100' /> | 
					
						
							|  |  |  |                   ) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 <div className='p-1'> | 
					
						
							|  |  |  |                   { | 
					
						
							|  |  |  |                     options.map(option => ( | 
					
						
							|  |  |  |                       <div | 
					
						
							|  |  |  |                         key={option.key} | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |                         className='flex items-center justify-between px-3 h-8 rounded-lg hover:bg-gray-50 cursor-pointer text-sm text-gray-700' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |                         onClick={() => handleZoom(option.key)} | 
					
						
							|  |  |  |                       > | 
					
						
							|  |  |  |                         {option.text} | 
					
						
							| 
									
										
										
										
											2024-05-09 17:18:51 +08:00
										 |  |  |                         { | 
					
						
							|  |  |  |                           option.key === ZoomType.zoomToFit && ( | 
					
						
							|  |  |  |                             <ShortcutsName keys={[`${getKeyboardKeyNameBySystem('ctrl')}`, '1']} /> | 
					
						
							|  |  |  |                           ) | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                           option.key === ZoomType.zoomTo50 && ( | 
					
						
							|  |  |  |                             <ShortcutsName keys={['shift', '5']} /> | 
					
						
							|  |  |  |                           ) | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                           option.key === ZoomType.zoomTo100 && ( | 
					
						
							|  |  |  |                             <ShortcutsName keys={['shift', '1']} /> | 
					
						
							|  |  |  |                           ) | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |                       </div> | 
					
						
							|  |  |  |                     )) | 
					
						
							|  |  |  |                   } | 
					
						
							|  |  |  |                 </div> | 
					
						
							|  |  |  |               </Fragment> | 
					
						
							|  |  |  |             )) | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       </PortalToFollowElemContent> | 
					
						
							|  |  |  |     </PortalToFollowElem> | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default memo(ZoomInOut) |