index.tsx 4.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. import { CheckCircleIcon } from '@heroicons/react/24/solid'
  2. import { QuestionMarkCircleIcon, XMarkIcon } from '@heroicons/react/24/outline'
  3. import { useTranslation } from 'react-i18next'
  4. import { useMemo } from 'react'
  5. import InvitationLink from './invitation-link'
  6. import s from './index.module.css'
  7. import Modal from '@/app/components/base/modal'
  8. import Button from '@/app/components/base/button'
  9. import { IS_CE_EDITION } from '@/config'
  10. import type { InvitationResult } from '@/models/common'
  11. import Tooltip from '@/app/components/base/tooltip'
  12. export type SuccessInvationResult = Extract<InvitationResult, { status: 'success' }>
  13. export type FailedInvationResult = Extract<InvitationResult, { status: 'failed' }>
  14. type IInvitedModalProps = {
  15. invitationResults: InvitationResult[]
  16. onCancel: () => void
  17. }
  18. const InvitedModal = ({
  19. invitationResults,
  20. onCancel,
  21. }: IInvitedModalProps) => {
  22. const { t } = useTranslation()
  23. const successInvationResults = useMemo<SuccessInvationResult[]>(() => invitationResults?.filter(item => item.status === 'success') as SuccessInvationResult[], [invitationResults])
  24. const failedInvationResults = useMemo<FailedInvationResult[]>(() => invitationResults?.filter(item => item.status !== 'success') as FailedInvationResult[], [invitationResults])
  25. return (
  26. <div className={s.wrap}>
  27. <Modal isShow onClose={() => {}} className={s.modal} wrapperClassName='z-20'>
  28. <div className='flex justify-between mb-3'>
  29. <div className='
  30. w-12 h-12 flex items-center justify-center rounded-xl
  31. bg-white border-[0.5px] border-gray-100
  32. shadow-xl
  33. '>
  34. <CheckCircleIcon className='w-[22px] h-[22px] text-[#039855]' />
  35. </div>
  36. <XMarkIcon className='w-4 h-4 cursor-pointer' onClick={onCancel} />
  37. </div>
  38. <div className='mb-1 text-xl font-semibold text-gray-900'>{t('common.members.invitationSent')}</div>
  39. {!IS_CE_EDITION && (
  40. <div className='mb-10 text-sm text-gray-500'>{t('common.members.invitationSentTip')}</div>
  41. )}
  42. {IS_CE_EDITION && (
  43. <>
  44. <div className='mb-5 text-sm text-gray-500'>{t('common.members.invitationSentTip')}</div>
  45. <div className='flex flex-col gap-2 mb-9'>
  46. {
  47. !!successInvationResults.length
  48. && <>
  49. <div className='py-2 text-sm font-Medium text-gray-900'>{t('common.members.invitationLink')}</div>
  50. {successInvationResults.map(item =>
  51. <InvitationLink key={item.email} value={item} />)}
  52. </>
  53. }
  54. {
  55. !!failedInvationResults.length
  56. && <>
  57. <div className='py-2 text-sm font-Medium text-gray-900'>{t('common.members.failedinvitationEmails')}</div>
  58. <div className='flex flex-wrap justify-between gap-y-1'>
  59. {
  60. failedInvationResults.map(item =>
  61. <div key={item.email} className='flex justify-center border border-red-300 rounded-md px-1 bg-orange-50'>
  62. <Tooltip
  63. selector={`invitation-tag-${item.email}`}
  64. htmlContent={item.message}
  65. >
  66. <div className='flex justify-center items-center text-sm gap-1'>
  67. {item.email}
  68. <QuestionMarkCircleIcon className='w-4 h-4 text-red-300' />
  69. </div>
  70. </Tooltip>
  71. </div>,
  72. )
  73. }
  74. </div>
  75. </>
  76. }
  77. </div>
  78. </>
  79. )}
  80. <div className='flex justify-end'>
  81. <Button
  82. className='w-[96px] text-sm font-medium'
  83. onClick={onCancel}
  84. type='primary'
  85. >
  86. {t('common.members.ok')}
  87. </Button>
  88. </div>
  89. </Modal>
  90. </div>
  91. )
  92. }
  93. export default InvitedModal