| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  | import type { FC } from 'react' | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  | import { useRef } from 'react' | 
					
						
							|  |  |  | import { t } from 'i18next' | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  | import { createPortal } from 'react-dom' | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  | import { RiCloseLine, RiExternalLinkLine } from '@remixicon/react' | 
					
						
							|  |  |  | import Tooltip from '@/app/components/base/tooltip' | 
					
						
							|  |  |  | import { randomString } from '@/utils' | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | type ImagePreviewProps = { | 
					
						
							|  |  |  |   url: string | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  |   title: string | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   onCancel: () => void | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | const ImagePreview: FC<ImagePreviewProps> = ({ | 
					
						
							|  |  |  |   url, | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  |   title, | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   onCancel, | 
					
						
							|  |  |  | }) => { | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  |   const selector = useRef(`copy-tooltip-${randomString(4)}`) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const openInNewTab = () => { | 
					
						
							|  |  |  |     // Open in a new window, considering the case when the page is inside an iframe
 | 
					
						
							|  |  |  |     if (url.startsWith('http')) { | 
					
						
							|  |  |  |       window.open(url, '_blank') | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (url.startsWith('data:image')) { | 
					
						
							|  |  |  |       // Base64 image
 | 
					
						
							|  |  |  |       const win = window.open() | 
					
						
							|  |  |  |       win?.document.write(`<img src="${url}" alt="${title}" />`) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       console.error('Unable to open image', url) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   return createPortal( | 
					
						
							|  |  |  |     <div className='fixed inset-0 p-8 flex items-center justify-center bg-black/80 z-[1000]' onClick={e => e.stopPropagation()}> | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  |       {/* eslint-disable-next-line @next/next/no-img-element */} | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |       <img | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  |         alt={title} | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |         src={url} | 
					
						
							|  |  |  |         className='max-w-full max-h-full' | 
					
						
							|  |  |  |       /> | 
					
						
							|  |  |  |       <div | 
					
						
							| 
									
										
										
										
											2024-06-05 00:13:29 +08:00
										 |  |  |         className='absolute top-6 right-6 flex items-center justify-center w-8 h-8 bg-white/8 rounded-lg backdrop-blur-[2px] cursor-pointer' | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |         onClick={onCancel} | 
					
						
							|  |  |  |       > | 
					
						
							| 
									
										
										
										
											2024-06-20 11:05:08 +08:00
										 |  |  |         <RiCloseLine className='w-4 h-4 text-white' /> | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |       </div> | 
					
						
							| 
									
										
										
										
											2024-09-05 21:00:09 +08:00
										 |  |  |       <Tooltip | 
					
						
							|  |  |  |         selector={selector.current} | 
					
						
							|  |  |  |         content={(t('common.operation.openInNewTab') ?? 'Open in new tab')} | 
					
						
							|  |  |  |         className='z-10' | 
					
						
							|  |  |  |       > | 
					
						
							|  |  |  |         <div | 
					
						
							|  |  |  |           className='absolute top-6 right-16 flex items-center justify-center w-8 h-8 rounded-lg cursor-pointer' | 
					
						
							|  |  |  |           onClick={openInNewTab} | 
					
						
							|  |  |  |         > | 
					
						
							|  |  |  |           <RiExternalLinkLine className='w-4 h-4 text-white' /> | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       </Tooltip> | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |     </div>, | 
					
						
							|  |  |  |     document.body, | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default ImagePreview |