organizationMemberRow.tsx 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. import {Fragment, PureComponent} from 'react';
  2. import styled from '@emotion/styled';
  3. import UserAvatar from 'sentry/components/avatar/userAvatar';
  4. import {Button} from 'sentry/components/button';
  5. import Confirm from 'sentry/components/confirm';
  6. import HookOrDefault from 'sentry/components/hookOrDefault';
  7. import Link from 'sentry/components/links/link';
  8. import LoadingIndicator from 'sentry/components/loadingIndicator';
  9. import PanelItem from 'sentry/components/panels/panelItem';
  10. import {IconCheckmark, IconClose, IconFlag, IconMail, IconSubtract} from 'sentry/icons';
  11. import {t, tct} from 'sentry/locale';
  12. import {space} from 'sentry/styles/space';
  13. import type {Member, Organization} from 'sentry/types/organization';
  14. import type {AvatarUser} from 'sentry/types/user';
  15. import isMemberDisabledFromLimit from 'sentry/utils/isMemberDisabledFromLimit';
  16. import {capitalize} from 'sentry/utils/string/capitalize';
  17. type Props = {
  18. canAddMembers: boolean;
  19. canRemoveMembers: boolean;
  20. currentUser: AvatarUser;
  21. member: Member;
  22. memberCanLeave: boolean;
  23. onLeave: (member: Member) => void;
  24. onRemove: (member: Member) => void;
  25. onSendInvite: (member: Member) => void;
  26. organization: Organization;
  27. requireLink: boolean;
  28. status: '' | 'loading' | 'success' | 'error' | null;
  29. };
  30. type State = {
  31. busy: boolean;
  32. };
  33. const DisabledMemberTooltip = HookOrDefault({
  34. hookName: 'component:disabled-member-tooltip',
  35. defaultComponent: ({children}) => <Fragment>{children}</Fragment>,
  36. });
  37. export default class OrganizationMemberRow extends PureComponent<Props, State> {
  38. state: State = {
  39. busy: false,
  40. };
  41. handleRemove = () => {
  42. const {onRemove} = this.props;
  43. if (typeof onRemove !== 'function') {
  44. return;
  45. }
  46. this.setState({busy: true});
  47. onRemove(this.props.member);
  48. };
  49. handleLeave = () => {
  50. const {onLeave} = this.props;
  51. if (typeof onLeave !== 'function') {
  52. return;
  53. }
  54. this.setState({busy: true});
  55. onLeave(this.props.member);
  56. };
  57. handleSendInvite = () => {
  58. const {onSendInvite, member} = this.props;
  59. if (typeof onSendInvite !== 'function') {
  60. return;
  61. }
  62. onSendInvite(member);
  63. };
  64. renderMemberRole() {
  65. const {member} = this.props;
  66. const {roleName, pending, expired} = member;
  67. if (isMemberDisabledFromLimit(member)) {
  68. return <DisabledMemberTooltip>{t('Deactivated')}</DisabledMemberTooltip>;
  69. }
  70. if (pending) {
  71. return (
  72. <InvitedRole>
  73. <IconMail size="md" />
  74. {expired ? t('Expired Invite') : tct('Invited [roleName]', {roleName})}
  75. </InvitedRole>
  76. );
  77. }
  78. return <Fragment>{capitalize(member.orgRole)}</Fragment>;
  79. }
  80. render() {
  81. const {
  82. member,
  83. organization,
  84. status,
  85. requireLink,
  86. memberCanLeave,
  87. currentUser,
  88. canRemoveMembers,
  89. canAddMembers,
  90. } = this.props;
  91. const {id, flags, email, name, pending, user, inviterName} = member;
  92. const {access} = organization;
  93. // if member is not the only owner, they can leave
  94. const isIdpProvisioned = flags['idp:provisioned'];
  95. const isPartnershipUser = flags['partnership:restricted'];
  96. const needsSso = !flags['sso:linked'] && requireLink;
  97. const isCurrentUser = currentUser.email === email;
  98. const showRemoveButton = !isCurrentUser;
  99. const showLeaveButton = isCurrentUser;
  100. const isInviteFromCurrentUser = pending && inviterName === currentUser.name;
  101. const canInvite = organization.allowMemberInvite && access.includes('member:invite');
  102. // members can remove invites they sent if allowMemberInvite is true
  103. const canEditInvite = canInvite && isInviteFromCurrentUser;
  104. const canRemoveMember =
  105. (canRemoveMembers && !isCurrentUser && !isIdpProvisioned && !isPartnershipUser) ||
  106. canEditInvite;
  107. // member has a `user` property if they are registered with sentry
  108. // i.e. has accepted an invite to join org
  109. const has2fa = user?.has2fa;
  110. const detailsUrl = `/settings/${organization.slug}/members/${id}/`;
  111. const isInviteSuccessful = status === 'success';
  112. const isInviting = status === 'loading';
  113. const showResendButton = pending || needsSso;
  114. return (
  115. <StyledPanelItem data-test-id={email}>
  116. <MemberHeading>
  117. <UserAvatar
  118. size={32}
  119. user={user ?? {email, id: email, name: email, type: 'user'}}
  120. />
  121. <MemberDescription to={detailsUrl}>
  122. <h5 style={{margin: '0 0 3px'}}>
  123. <UserName>{name}</UserName>
  124. </h5>
  125. <Email>{email}</Email>
  126. </MemberDescription>
  127. </MemberHeading>
  128. <div data-test-id="member-role">{this.renderMemberRole()}</div>
  129. <div data-test-id="member-status">
  130. {showResendButton ? (
  131. <Fragment>
  132. {isInviting && (
  133. <LoadingContainer>
  134. <LoadingIndicator mini />
  135. </LoadingContainer>
  136. )}
  137. {isInviteSuccessful && <span>{t('Sent!')}</span>}
  138. {!isInviting && !isInviteSuccessful && (
  139. <Button
  140. disabled={!canAddMembers && !canEditInvite}
  141. priority="primary"
  142. size="sm"
  143. onClick={this.handleSendInvite}
  144. >
  145. {pending ? t('Resend invite') : t('Resend SSO link')}
  146. </Button>
  147. )}
  148. </Fragment>
  149. ) : (
  150. <AuthStatus>
  151. {has2fa ? (
  152. <IconCheckmark isCircled color="success" />
  153. ) : (
  154. <IconFlag color="error" />
  155. )}
  156. {has2fa ? t('2FA Enabled') : t('2FA Not Enabled')}
  157. </AuthStatus>
  158. )}
  159. </div>
  160. {showRemoveButton || showLeaveButton ? (
  161. <RightColumn>
  162. {showRemoveButton && canRemoveMember && (
  163. <Confirm
  164. message={tct('Are you sure you want to remove [name] from [orgName]?', {
  165. name,
  166. orgName: organization.slug,
  167. })}
  168. onConfirm={this.handleRemove}
  169. >
  170. <Button
  171. data-test-id="remove"
  172. icon={<IconSubtract isCircled />}
  173. size="sm"
  174. busy={this.state.busy}
  175. >
  176. {t('Remove')}
  177. </Button>
  178. </Confirm>
  179. )}
  180. {showRemoveButton && !canRemoveMember && (
  181. <Button
  182. disabled
  183. size="sm"
  184. title={
  185. isIdpProvisioned
  186. ? t(
  187. "This user is managed through your organization's identity provider."
  188. )
  189. : isPartnershipUser
  190. ? t('You cannot make changes to this partner-provisioned user.')
  191. : // only show this message if member can remove invites but invite was not sent by them
  192. pending && canInvite && !isInviteFromCurrentUser
  193. ? t('You cannot modify this invite.')
  194. : t('You do not have access to remove members')
  195. }
  196. icon={<IconSubtract isCircled />}
  197. >
  198. {t('Remove')}
  199. </Button>
  200. )}
  201. {showLeaveButton && memberCanLeave && (
  202. <Confirm
  203. message={tct('Are you sure you want to leave [orgName]?', {
  204. orgName: organization.slug,
  205. })}
  206. onConfirm={this.handleLeave}
  207. >
  208. <Button priority="danger" size="sm" icon={<IconClose />}>
  209. {t('Leave')}
  210. </Button>
  211. </Confirm>
  212. )}
  213. {showLeaveButton && !memberCanLeave && (
  214. <Button
  215. size="sm"
  216. icon={<IconClose />}
  217. disabled
  218. title={
  219. isIdpProvisioned
  220. ? t(
  221. "Your account is managed through your organization's identity provider."
  222. )
  223. : isPartnershipUser
  224. ? t('You cannot make changes as a partner-provisioned user.')
  225. : t(
  226. 'You cannot leave this organization as you are the only organization owner.'
  227. )
  228. }
  229. >
  230. {t('Leave')}
  231. </Button>
  232. )}
  233. </RightColumn>
  234. ) : null}
  235. </StyledPanelItem>
  236. );
  237. }
  238. }
  239. const StyledPanelItem = styled(PanelItem)`
  240. display: grid;
  241. grid-template-columns: minmax(150px, 4fr) minmax(90px, 2fr) minmax(120px, 2fr) minmax(
  242. 100px,
  243. 1fr
  244. );
  245. gap: ${space(2)};
  246. align-items: center;
  247. `;
  248. // Force action button at the end to align to right
  249. const RightColumn = styled('div')`
  250. display: flex;
  251. justify-content: flex-end;
  252. `;
  253. const Section = styled('div')`
  254. display: inline-grid;
  255. grid-template-columns: max-content auto;
  256. gap: ${space(1)};
  257. align-items: center;
  258. `;
  259. const MemberHeading = styled(Section)``;
  260. const MemberDescription = styled(Link)`
  261. overflow: hidden;
  262. `;
  263. const UserName = styled('div')`
  264. display: block;
  265. overflow: hidden;
  266. font-size: ${p => p.theme.fontSizeMedium};
  267. text-overflow: ellipsis;
  268. `;
  269. const Email = styled('div')`
  270. color: ${p => p.theme.subText};
  271. font-size: ${p => p.theme.fontSizeSmall};
  272. overflow: hidden;
  273. text-overflow: ellipsis;
  274. `;
  275. const InvitedRole = styled(Section)``;
  276. const LoadingContainer = styled('div')`
  277. margin-top: 0;
  278. margin-bottom: ${space(1.5)};
  279. `;
  280. const AuthStatus = styled(Section)``;