| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182 | 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
 |