mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-26 08:28:55 +00:00 
			
		
		
		
	 7709d9df20
			
		
	
	
		7709d9df20
		
			
		
	
	
	
	
		
			
			Co-authored-by: NFish <douxc512@gmail.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: jZonG <jzongcode@gmail.com>
		
			
				
	
	
		
			186 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			186 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { Menu, MenuButton, MenuItem, MenuItems, Transition } from '@headlessui/react'
 | |
| import { RiArrowDownCircleLine, RiArrowRightSLine, RiVerifiedBadgeLine } from '@remixicon/react'
 | |
| import type { FC, MouseEvent } from 'react'
 | |
| import { Fragment, useCallback } from 'react'
 | |
| import { useTranslation } from 'react-i18next'
 | |
| import { useMutation } from '@tanstack/react-query'
 | |
| import PremiumBadge from '../../base/premium-badge'
 | |
| import SparklesSoft from '../../base/icons/src/public/common/SparklesSoft'
 | |
| import Button from '../../base/button'
 | |
| import Soc2 from '../../base/icons/src/public/common/Soc2'
 | |
| import Iso from '../../base/icons/src/public/common/Iso'
 | |
| import Gdpr from '../../base/icons/src/public/common/Gdpr'
 | |
| import Toast from '../../base/toast'
 | |
| import Tooltip from '../../base/tooltip'
 | |
| import cn from '@/utils/classnames'
 | |
| import { useProviderContext } from '@/context/provider-context'
 | |
| import { Plan } from '@/app/components/billing/type'
 | |
| import { useModalContext } from '@/context/modal-context'
 | |
| import { getDocDownloadUrl } from '@/service/common'
 | |
| 
 | |
| enum DocName {
 | |
|   SOC2_Type_I = 'SOC2_Type_I',
 | |
|   SOC2_Type_II = 'SOC2_Type_II',
 | |
|   ISO_27001 = 'ISO_27001',
 | |
|   GDPR = 'GDPR',
 | |
| }
 | |
| 
 | |
| type UpgradeOrDownloadProps = {
 | |
|   doc_name: DocName
 | |
| }
 | |
| const UpgradeOrDownload: FC<UpgradeOrDownloadProps> = ({ doc_name }) => {
 | |
|   const { t } = useTranslation()
 | |
|   const { plan } = useProviderContext()
 | |
|   const { setShowPricingModal, setShowAccountSettingModal } = useModalContext()
 | |
|   const isFreePlan = plan.type === Plan.sandbox
 | |
| 
 | |
|   const handlePlanClick = useCallback(() => {
 | |
|     if (isFreePlan)
 | |
|       setShowPricingModal()
 | |
|     else
 | |
|       setShowAccountSettingModal({ payload: 'billing' })
 | |
|   }, [isFreePlan, setShowAccountSettingModal, setShowPricingModal])
 | |
| 
 | |
|   const { isPending, mutate: downloadCompliance } = useMutation({
 | |
|     mutationKey: ['downloadCompliance', doc_name],
 | |
|     mutationFn: async () => {
 | |
|       try {
 | |
|         const ret = await getDocDownloadUrl(doc_name)
 | |
|         const a = document.createElement('a')
 | |
|         a.href = ret.url
 | |
|         a.click()
 | |
|         Toast.notify({
 | |
|           type: 'success',
 | |
|           message: t('common.operation.downloadSuccess'),
 | |
|         })
 | |
|       }
 | |
|       catch (error) {
 | |
|         console.error(error)
 | |
|         Toast.notify({
 | |
|           type: 'error',
 | |
|           message: t('common.operation.downloadFailed'),
 | |
|         })
 | |
|       }
 | |
|     },
 | |
|   })
 | |
|   const whichPlanCanDownloadCompliance = {
 | |
|     [DocName.SOC2_Type_I]: [Plan.professional, Plan.team],
 | |
|     [DocName.SOC2_Type_II]: [Plan.team],
 | |
|     [DocName.ISO_27001]: [Plan.team],
 | |
|     [DocName.GDPR]: [Plan.team, Plan.professional, Plan.sandbox],
 | |
|   }
 | |
| 
 | |
|   const isCurrentPlanCanDownload = whichPlanCanDownloadCompliance[doc_name].includes(plan.type)
 | |
|   const handleDownloadClick = useCallback((e: MouseEvent<HTMLButtonElement>) => {
 | |
|     e.preventDefault()
 | |
|     downloadCompliance()
 | |
|   }, [downloadCompliance])
 | |
|   if (isCurrentPlanCanDownload) {
 | |
|     return <Button loading={isPending} disabled={isPending} size='small' variant='secondary' className='flex items-center gap-[1px]' onClick={handleDownloadClick}>
 | |
|       <RiArrowDownCircleLine className='size-[14px] text-components-button-secondary-text-disabled' />
 | |
|       <span className='system-xs-medium px-[3px] text-components-button-secondary-text'>{t('common.operation.download')}</span>
 | |
|     </Button>
 | |
|   }
 | |
|   const upgradeTooltip: Record<Plan, string> = {
 | |
|     [Plan.sandbox]: t('common.compliance.sandboxUpgradeTooltip'),
 | |
|     [Plan.professional]: t('common.compliance.professionalUpgradeTooltip'),
 | |
|     [Plan.team]: '',
 | |
|     [Plan.enterprise]: '',
 | |
|   }
 | |
|   return <Tooltip asChild={false} popupContent={upgradeTooltip[plan.type]}>
 | |
|     <PremiumBadge color='blue' allowHover={true} onClick={handlePlanClick}>
 | |
|       <SparklesSoft className='flex h-3.5 w-3.5 items-center py-[1px] pl-[3px] text-components-premium-badge-indigo-text-stop-0' />
 | |
|       <div className='system-xs-medium'>
 | |
|         <span className='p-1'>
 | |
|           {t('billing.upgradeBtn.encourageShort')}
 | |
|         </span>
 | |
|       </div>
 | |
|     </PremiumBadge>
 | |
|   </Tooltip>
 | |
| }
 | |
| 
 | |
| export default function Compliance() {
 | |
|   const itemClassName = `
 | |
|   flex items-center w-full h-10 pl-1 pr-2 py-1 text-text-secondary system-md-regular
 | |
|   rounded-lg hover:bg-state-base-hover gap-1
 | |
| `
 | |
|   const { t } = useTranslation()
 | |
| 
 | |
|   return <Menu as="div" className="relative h-full w-full">
 | |
|     {
 | |
|       ({ open }) => (
 | |
|         <>
 | |
|           <MenuButton className={
 | |
|             cn('group flex h-9 w-full items-center gap-1 rounded-lg py-2 pl-3 pr-2 hover:bg-state-base-hover',
 | |
|               open && 'bg-state-base-hover',
 | |
|             )}>
 | |
|             <RiVerifiedBadgeLine className='size-4 shrink-0 text-text-tertiary' />
 | |
|             <div className='system-md-regular grow px-1 text-left text-text-secondary'>{t('common.userProfile.compliance')}</div>
 | |
|             <RiArrowRightSLine className='size-[14px] shrink-0 text-text-tertiary' />
 | |
|           </MenuButton>
 | |
|           <Transition
 | |
|             as={Fragment}
 | |
|             enter="transition ease-out duration-100"
 | |
|             enterFrom="transform opacity-0 scale-95"
 | |
|             enterTo="transform opacity-100 scale-100"
 | |
|             leave="transition ease-in duration-75"
 | |
|             leaveFrom="transform opacity-100 scale-100"
 | |
|             leaveTo="transform opacity-0 scale-95"
 | |
|           >
 | |
|             <MenuItems
 | |
|               className={cn(
 | |
|                 `absolute top-[1px] z-10 max-h-[70vh] w-[337px] origin-top-right -translate-x-full divide-y divide-divider-subtle overflow-y-scroll
 | |
|                 rounded-xl border-[0.5px] border-components-panel-border bg-components-panel-bg-blur shadow-lg backdrop-blur-[5px] focus:outline-none
 | |
|               `,
 | |
|               )}
 | |
|             >
 | |
|               <div className="px-1 py-1">
 | |
|                 <MenuItem>
 | |
|                   <div
 | |
|                     className={cn(itemClassName, 'group justify-between',
 | |
|                       'data-[active]:bg-state-base-hover',
 | |
|                     )}>
 | |
|                     <Soc2 className='size-7 shrink-0' />
 | |
|                     <div className='system-md-regular grow truncate px-1 text-text-secondary'>{t('common.compliance.soc2Type1')}</div>
 | |
|                     <UpgradeOrDownload doc_name={DocName.SOC2_Type_I} />
 | |
|                   </div>
 | |
|                 </MenuItem>
 | |
|                 <MenuItem>
 | |
|                   <div
 | |
|                     className={cn(itemClassName, 'group justify-between',
 | |
|                       'data-[active]:bg-state-base-hover',
 | |
|                     )}>
 | |
|                     <Soc2 className='size-7 shrink-0' />
 | |
|                     <div className='system-md-regular grow truncate px-1 text-text-secondary'>{t('common.compliance.soc2Type2')}</div>
 | |
|                     <UpgradeOrDownload doc_name={DocName.SOC2_Type_II} />
 | |
|                   </div>
 | |
|                 </MenuItem>
 | |
|                 <MenuItem>
 | |
|                   <div
 | |
|                     className={cn(itemClassName, 'group justify-between',
 | |
|                       'data-[active]:bg-state-base-hover',
 | |
|                     )}>
 | |
|                     <Iso className='size-7 shrink-0' />
 | |
|                     <div className='system-md-regular grow truncate px-1 text-text-secondary'>{t('common.compliance.iso27001')}</div>
 | |
|                     <UpgradeOrDownload doc_name={DocName.ISO_27001} />
 | |
|                   </div>
 | |
|                 </MenuItem>
 | |
|                 <MenuItem>
 | |
|                   <div
 | |
|                     className={cn(itemClassName, 'group justify-between',
 | |
|                       'data-[active]:bg-state-base-hover',
 | |
|                     )}>
 | |
|                     <Gdpr className='size-7 shrink-0' />
 | |
|                     <div className='system-md-regular grow truncate px-1 text-text-secondary'>{t('common.compliance.gdpr')}</div>
 | |
|                     <UpgradeOrDownload doc_name={DocName.GDPR} />
 | |
|                   </div>
 | |
|                 </MenuItem>
 | |
|               </div>
 | |
|             </MenuItems>
 | |
|           </Transition>
 | |
|         </>
 | |
|       )
 | |
|     }
 | |
|   </Menu>
 | |
| }
 |