|
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 Input from '@/app/components/base/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] rounded-lg border-[0.5px] bg-white 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'> |
|
<Input |
|
showLeftIcon |
|
showClearIcon |
|
value={keywords} |
|
onChange={e => handleKeywordsChange(e.target.value)} |
|
onClear={() => 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 |
|
|