Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

index.tsx 4.0KB

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