inviteRequestRow.tsx 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. import {Fragment} from 'react';
  2. import styled from '@emotion/styled';
  3. import Button from 'sentry/components/button';
  4. import Confirm from 'sentry/components/confirm';
  5. import {MultiControlProps} from 'sentry/components/deprecatedforms/multiSelectControl';
  6. import TeamSelector from 'sentry/components/forms/teamSelector';
  7. import HookOrDefault from 'sentry/components/hookOrDefault';
  8. import {PanelItem} from 'sentry/components/panels';
  9. import RoleSelectControl from 'sentry/components/roleSelectControl';
  10. import Tag from 'sentry/components/tag';
  11. import Tooltip from 'sentry/components/tooltip';
  12. import {IconCheckmark, IconClose} from 'sentry/icons';
  13. import {t, tct} from 'sentry/locale';
  14. import space from 'sentry/styles/space';
  15. import {Member, Organization, OrgRole} from 'sentry/types';
  16. type Props = {
  17. allRoles: OrgRole[];
  18. inviteRequest: Member;
  19. inviteRequestBusy: {[key: string]: boolean};
  20. onApprove: (inviteRequest: Member) => void;
  21. onDeny: (inviteRequest: Member) => void;
  22. onUpdate: (data: Partial<Member>) => void;
  23. organization: Organization;
  24. };
  25. const InviteModalHook = HookOrDefault({
  26. hookName: 'member-invite-modal:customization',
  27. defaultComponent: ({onSendInvites, children}) =>
  28. children({sendInvites: onSendInvites, canSend: true}),
  29. });
  30. type InviteModalRenderFunc = React.ComponentProps<typeof InviteModalHook>['children'];
  31. type OnChangeArgs = Parameters<NonNullable<MultiControlProps['onChange']>>[0];
  32. const InviteRequestRow = ({
  33. inviteRequest,
  34. inviteRequestBusy,
  35. organization,
  36. onApprove,
  37. onDeny,
  38. onUpdate,
  39. allRoles,
  40. }: Props) => {
  41. const role = allRoles.find(r => r.id === inviteRequest.role);
  42. const roleDisallowed = !(role && role.allowed);
  43. const {access} = organization;
  44. const canApprove = access.includes('member:admin');
  45. // eslint-disable-next-line react/prop-types
  46. const hookRenderer: InviteModalRenderFunc = ({sendInvites, canSend, headerInfo}) => (
  47. <StyledPanelItem>
  48. <div>
  49. <h5 style={{marginBottom: space(0.5)}}>
  50. <UserName>{inviteRequest.email}</UserName>
  51. </h5>
  52. {inviteRequest.inviteStatus === 'requested_to_be_invited' ? (
  53. inviteRequest.inviterName && (
  54. <Description>
  55. <Tooltip
  56. title={t(
  57. 'An existing member has asked to invite this user to your organization'
  58. )}
  59. >
  60. {tct('Requested by [inviterName]', {
  61. inviterName: inviteRequest.inviterName,
  62. })}
  63. </Tooltip>
  64. </Description>
  65. )
  66. ) : (
  67. <JoinRequestIndicator
  68. tooltipText={t('This user has asked to join your organization.')}
  69. >
  70. {t('Join request')}
  71. </JoinRequestIndicator>
  72. )}
  73. </div>
  74. {canApprove ? (
  75. <StyledRoleSelectControl
  76. name="role"
  77. disableUnallowed
  78. onChange={r => onUpdate({role: r.value})}
  79. value={inviteRequest.role}
  80. roles={allRoles}
  81. />
  82. ) : (
  83. <div>{inviteRequest.roleName}</div>
  84. )}
  85. {canApprove ? (
  86. <TeamSelectControl
  87. name="teams"
  88. placeholder={t('Add to teams\u2026')}
  89. onChange={(teams: OnChangeArgs) =>
  90. onUpdate({teams: (teams || []).map(team => team.value)})
  91. }
  92. value={inviteRequest.teams}
  93. clearable
  94. multiple
  95. />
  96. ) : (
  97. <div>{inviteRequest.teams.join(', ')}</div>
  98. )}
  99. <ButtonGroup>
  100. <Button
  101. size="sm"
  102. busy={inviteRequestBusy[inviteRequest.id]}
  103. onClick={() => onDeny(inviteRequest)}
  104. icon={<IconClose />}
  105. disabled={!canApprove}
  106. title={
  107. canApprove
  108. ? undefined
  109. : t('This request needs to be reviewed by a privileged user')
  110. }
  111. >
  112. {t('Deny')}
  113. </Button>
  114. <Confirm
  115. onConfirm={sendInvites}
  116. disableConfirmButton={!canSend}
  117. disabled={!canApprove || roleDisallowed}
  118. message={
  119. <Fragment>
  120. {tct('Are you sure you want to invite [email] to your organization?', {
  121. email: inviteRequest.email,
  122. })}
  123. {headerInfo}
  124. </Fragment>
  125. }
  126. >
  127. <Button
  128. priority="primary"
  129. size="sm"
  130. busy={inviteRequestBusy[inviteRequest.id]}
  131. title={
  132. canApprove
  133. ? roleDisallowed
  134. ? t(
  135. `You do not have permission to approve a user of this role.
  136. Select a different role to approve this user.`
  137. )
  138. : undefined
  139. : t('This request needs to be reviewed by a privileged user')
  140. }
  141. icon={<IconCheckmark />}
  142. >
  143. {t('Approve')}
  144. </Button>
  145. </Confirm>
  146. </ButtonGroup>
  147. </StyledPanelItem>
  148. );
  149. return (
  150. <InviteModalHook
  151. willInvite
  152. organization={organization}
  153. onSendInvites={() => onApprove(inviteRequest)}
  154. >
  155. {hookRenderer}
  156. </InviteModalHook>
  157. );
  158. };
  159. const JoinRequestIndicator = styled(Tag)`
  160. text-transform: uppercase;
  161. `;
  162. const StyledPanelItem = styled(PanelItem)`
  163. display: grid;
  164. grid-template-columns: minmax(150px, auto) minmax(100px, 140px) 220px max-content;
  165. gap: ${space(2)};
  166. align-items: center;
  167. `;
  168. const UserName = styled('div')`
  169. font-size: ${p => p.theme.fontSizeLarge};
  170. overflow: hidden;
  171. text-overflow: ellipsis;
  172. `;
  173. const Description = styled('div')`
  174. display: block;
  175. color: ${p => p.theme.subText};
  176. font-size: 14px;
  177. overflow: hidden;
  178. text-overflow: ellipsis;
  179. `;
  180. const StyledRoleSelectControl = styled(RoleSelectControl)`
  181. max-width: 140px;
  182. `;
  183. const TeamSelectControl = styled(TeamSelector)`
  184. max-width: 220px;
  185. .Select-value-label {
  186. max-width: 150px;
  187. word-break: break-all;
  188. }
  189. `;
  190. const ButtonGroup = styled('div')`
  191. display: inline-grid;
  192. grid-template-columns: repeat(2, max-content);
  193. gap: ${space(1)};
  194. `;
  195. export default InviteRequestRow;