| 
                        123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 | 
                        - import { useTranslation } from 'react-i18next'
 - import cn from 'classnames'
 - import React, { useMemo, useState } from 'react'
 - import { useDebounceFn } from 'ahooks'
 - import { RiArrowDownSLine } from '@remixicon/react'
 - import {
 -   PortalToFollowElem,
 -   PortalToFollowElemContent,
 -   PortalToFollowElemTrigger,
 - } from '@/app/components/base/portal-to-follow-elem'
 - import Avatar from '@/app/components/base/avatar'
 - import SearchInput from '@/app/components/base/search-input'
 - import { Check } from '@/app/components/base/icons/src/vender/line/general'
 - import { Users01, UsersPlus } from '@/app/components/base/icons/src/vender/solid/users'
 - import type { DatasetPermission } from '@/models/datasets'
 - import { useAppContext } from '@/context/app-context'
 - import type { Member } from '@/models/common'
 - export type RoleSelectorProps = {
 -   disabled?: boolean
 -   permission?: DatasetPermission
 -   value: string[]
 -   memberList: Member[]
 -   onChange: (permission?: DatasetPermission) => void
 -   onMemberSelect: (v: string[]) => void
 - }
 - 
 - const PermissionSelector = ({ disabled, permission, value, memberList, onChange, onMemberSelect }: RoleSelectorProps) => {
 -   const { t } = useTranslation()
 -   const { userProfile } = useAppContext()
 -   const [open, setOpen] = useState(false)
 - 
 -   const [keywords, setKeywords] = useState('')
 -   const [searchKeywords, setSearchKeywords] = useState('')
 -   const { run: handleSearch } = useDebounceFn(() => {
 -     setSearchKeywords(keywords)
 -   }, { wait: 500 })
 -   const handleKeywordsChange = (value: string) => {
 -     setKeywords(value)
 -     handleSearch()
 -   }
 -   const selectMember = (member: Member) => {
 -     if (value.includes(member.id))
 -       onMemberSelect(value.filter(v => v !== member.id))
 -     else
 -       onMemberSelect([...value, member.id])
 -   }
 - 
 -   const selectedMembers = useMemo(() => {
 -     return [
 -       userProfile,
 -       ...memberList.filter(member => member.id !== userProfile.id).filter(member => value.includes(member.id)),
 -     ].map(member => member.name).join(', ')
 -   }, [userProfile, value, memberList])
 - 
 -   const showMe = useMemo(() => {
 -     return userProfile.name.includes(searchKeywords) || userProfile.email.includes(searchKeywords)
 -   }, [searchKeywords, userProfile])
 - 
 -   const filteredMemberList = useMemo(() => {
 -     return memberList.filter(member => (member.name.includes(searchKeywords) || member.email.includes(searchKeywords)) && member.id !== userProfile.id && ['owner', 'admin', 'editor', 'dataset_operator'].includes(member.role))
 -   }, [memberList, searchKeywords, userProfile])
 - 
 -   return (
 -     <PortalToFollowElem
 -       open={open}
 -       onOpenChange={setOpen}
 -       placement='bottom-start'
 -       offset={4}
 -     >
 -       <div className='relative'>
 -         <PortalToFollowElemTrigger
 -           onClick={() => !disabled && setOpen(v => !v)}
 -           className='block'
 -         >
 -           {permission === 'only_me' && (
 -             <div className={cn('flex items-center px-3 py-[6px] rounded-lg bg-gray-100 cursor-pointer hover:bg-gray-200', open && 'bg-gray-200', disabled && 'hover:!bg-gray-100 !cursor-default')}>
 -               <Avatar name={userProfile.name} className='shrink-0 mr-2' size={24} />
 -               <div className='grow mr-2 text-gray-900 text-sm leading-5'>{t('datasetSettings.form.permissionsOnlyMe')}</div>
 -               {!disabled && <RiArrowDownSLine className='shrink-0 w-4 h-4 text-gray-700' />}
 -             </div>
 -           )}
 -           {permission === 'all_team_members' && (
 -             <div className={cn('flex items-center px-3 py-[6px] rounded-lg bg-gray-100 cursor-pointer hover:bg-gray-200', open && 'bg-gray-200')}>
 -               <div className='mr-2 flex items-center justify-center w-6 h-6 rounded-lg bg-[#EEF4FF]'>
 -                 <Users01 className='w-3.5 h-3.5 text-[#444CE7]' />
 -               </div>
 -               <div className='grow mr-2 text-gray-900 text-sm leading-5'>{t('datasetSettings.form.permissionsAllMember')}</div>
 -               {!disabled && <RiArrowDownSLine className='shrink-0 w-4 h-4 text-gray-700' />}
 -             </div>
 -           )}
 -           {permission === 'partial_members' && (
 -             <div className={cn('flex items-center px-3 py-[6px] rounded-lg bg-gray-100 cursor-pointer hover:bg-gray-200', open && 'bg-gray-200')}>
 -               <div className='mr-2 flex items-center justify-center w-6 h-6 rounded-lg bg-[#EEF4FF]'>
 -                 <Users01 className='w-3.5 h-3.5 text-[#444CE7]' />
 -               </div>
 -               <div title={selectedMembers} className='grow mr-2 text-gray-900 text-sm leading-5 truncate'>{selectedMembers}</div>
 -               {!disabled && <RiArrowDownSLine className='shrink-0 w-4 h-4 text-gray-700' />}
 -             </div>
 -           )}
 -         </PortalToFollowElemTrigger>
 -         <PortalToFollowElemContent className='z-[1002]'>
 -           <div className='relative w-[480px] bg-white rounded-lg border-[0.5px] bg-gray-200 shadow-lg'>
 -             <div className='p-1'>
 -               <div className='pl-3 pr-2 py-1 rounded-lg hover:bg-gray-50 cursor-pointer' onClick={() => {
 -                 onChange('only_me')
 -                 setOpen(false)
 -               }}>
 -                 <div className='flex items-center gap-2'>
 -                   <Avatar name={userProfile.name} className='shrink-0 mr-2' size={24} />
 -                   <div className='grow mr-2 text-gray-900 text-sm leading-5'>{t('datasetSettings.form.permissionsOnlyMe')}</div>
 -                   {permission === 'only_me' && <Check className='w-4 h-4 text-primary-600' />}
 -                 </div>
 -               </div>
 -               <div className='pl-3 pr-2 py-1 rounded-lg hover:bg-gray-50 cursor-pointer' onClick={() => {
 -                 onChange('all_team_members')
 -                 setOpen(false)
 -               }}>
 -                 <div className='flex items-center gap-2'>
 -                   <div className='mr-2 flex items-center justify-center w-6 h-6 rounded-lg bg-[#EEF4FF]'>
 -                     <Users01 className='w-3.5 h-3.5 text-[#444CE7]' />
 -                   </div>
 -                   <div className='grow mr-2 text-gray-900 text-sm leading-5'>{t('datasetSettings.form.permissionsAllMember')}</div>
 -                   {permission === 'all_team_members' && <Check className='w-4 h-4 text-primary-600' />}
 -                 </div>
 -               </div>
 -               <div className='pl-3 pr-2 py-1 rounded-lg hover:bg-gray-50 cursor-pointer' onClick={() => {
 -                 onChange('partial_members')
 -                 onMemberSelect([userProfile.id])
 -               }}>
 -                 <div className='flex items-center gap-2'>
 -                   <div className={cn('mr-2 flex items-center justify-center w-6 h-6 rounded-lg bg-[#FFF6ED]', permission === 'partial_members' && '!bg-[#EEF4FF]')}>
 -                     <UsersPlus className={cn('w-3.5 h-3.5 text-[#FB6514]', permission === 'partial_members' && '!text-[#444CE7]')} />
 -                   </div>
 -                   <div className='grow mr-2 text-gray-900 text-sm leading-5'>{t('datasetSettings.form.permissionsInvitedMembers')}</div>
 -                   {permission === 'partial_members' && <Check className='w-4 h-4 text-primary-600' />}
 -                 </div>
 -               </div>
 -             </div>
 -             {permission === 'partial_members' && (
 -               <div className='max-h-[360px] border-t-[1px] border-gray-100 p-1 overflow-y-auto'>
 -                 <div className='sticky left-0 top-0 p-2 pb-1 bg-white'>
 -                   <SearchInput white value={keywords} onChange={handleKeywordsChange} />
 -                 </div>
 -                 {showMe && (
 -                   <div className='pl-3 pr-[10px] py-1 flex gap-2 items-center rounded-lg'>
 -                     <Avatar name={userProfile.name} className='shrink-0' size={24} />
 -                     <div className='grow'>
 -                       <div className='text-[13px] text-gray-700 font-medium leading-[18px] truncate'>
 -                         {userProfile.name}
 -                         <span className='text-xs text-gray-500 font-normal'>{t('datasetSettings.form.me')}</span>
 -                       </div>
 -                       <div className='text-xs text-gray-500 leading-[18px] truncate'>{userProfile.email}</div>
 -                     </div>
 -                     <Check className='shrink-0 w-4 h-4 text-primary-600 opacity-30' />
 -                   </div>
 -                 )}
 -                 {filteredMemberList.map(member => (
 -                   <div key={member.id} className='pl-3 pr-[10px] py-1 flex gap-2 items-center rounded-lg hover:bg-gray-100 cursor-pointer' onClick={() => selectMember(member)}>
 -                     <Avatar name={member.name} className='shrink-0' size={24} />
 -                     <div className='grow'>
 -                       <div className='text-[13px] text-gray-700 font-medium leading-[18px] truncate'>{member.name}</div>
 -                       <div className='text-xs text-gray-500 leading-[18px] truncate'>{member.email}</div>
 -                     </div>
 -                     {value.includes(member.id) && <Check className='shrink-0 w-4 h-4 text-primary-600' />}
 -                   </div>
 -                 ))}
 -               </div>
 -             )}
 -           </div>
 -         </PortalToFollowElemContent>
 -       </div>
 -     </PortalToFollowElem>
 -   )
 - }
 - 
 - export default PermissionSelector
 
 
  |