teamMembers.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. import {Fragment} from 'react';
  2. import type {RouteComponentProps} from 'react-router';
  3. import styled from '@emotion/styled';
  4. import debounce from 'lodash/debounce';
  5. import {addErrorMessage, addSuccessMessage} from 'sentry/actionCreators/indicator';
  6. import {
  7. openInviteMembersModal,
  8. openTeamAccessRequestModal,
  9. } from 'sentry/actionCreators/modal';
  10. import {joinTeam, leaveTeam} from 'sentry/actionCreators/teams';
  11. import type {Client} from 'sentry/api';
  12. import {hasEveryAccess} from 'sentry/components/acl/access';
  13. import UserAvatar from 'sentry/components/avatar/userAvatar';
  14. import DropdownAutoComplete from 'sentry/components/dropdownAutoComplete';
  15. import type {Item} from 'sentry/components/dropdownAutoComplete/types';
  16. import DropdownButton from 'sentry/components/dropdownButton';
  17. import EmptyMessage from 'sentry/components/emptyMessage';
  18. import Link from 'sentry/components/links/link';
  19. import LoadingError from 'sentry/components/loadingError';
  20. import LoadingIndicator from 'sentry/components/loadingIndicator';
  21. import Pagination from 'sentry/components/pagination';
  22. import Panel from 'sentry/components/panels/panel';
  23. import PanelHeader from 'sentry/components/panels/panelHeader';
  24. import {TeamRoleColumnLabel} from 'sentry/components/teamRoleUtils';
  25. import {IconUser} from 'sentry/icons';
  26. import {t} from 'sentry/locale';
  27. import {space} from 'sentry/styles/space';
  28. import type {Config, Member, Organization, Team, TeamMember} from 'sentry/types';
  29. import withApi from 'sentry/utils/withApi';
  30. import withConfig from 'sentry/utils/withConfig';
  31. import withOrganization from 'sentry/utils/withOrganization';
  32. import type {AsyncViewState} from 'sentry/views/deprecatedAsyncView';
  33. import DeprecatedAsyncView from 'sentry/views/deprecatedAsyncView';
  34. import TextBlock from 'sentry/views/settings/components/text/textBlock';
  35. import TeamMembersRow, {
  36. GRID_TEMPLATE,
  37. } from 'sentry/views/settings/organizationTeams/teamMembersRow';
  38. import PermissionAlert from 'sentry/views/settings/project/permissionAlert';
  39. import {getButtonHelpText} from './utils';
  40. type RouteParams = {
  41. teamId: string;
  42. };
  43. interface Props extends RouteComponentProps<RouteParams, {}> {
  44. api: Client;
  45. config: Config;
  46. organization: Organization;
  47. team: Team;
  48. }
  49. interface State extends AsyncViewState {
  50. dropdownBusy: boolean;
  51. error: boolean;
  52. orgMembers: Member[];
  53. teamMembers: TeamMember[];
  54. }
  55. class TeamMembers extends DeprecatedAsyncView<Props, State> {
  56. getDefaultState() {
  57. return {
  58. ...super.getDefaultState(),
  59. error: false,
  60. dropdownBusy: false,
  61. teamMembers: [],
  62. orgMembers: [],
  63. };
  64. }
  65. componentDidMount() {
  66. super.componentDidMount();
  67. // Initialize "add member" dropdown with data
  68. this.fetchMembersRequest('');
  69. }
  70. debouncedFetchMembersRequest = debounce(
  71. (query: string) =>
  72. this.setState({dropdownBusy: true}, () => this.fetchMembersRequest(query)),
  73. 200
  74. );
  75. fetchMembersRequest = async (query: string) => {
  76. const {organization, api} = this.props;
  77. try {
  78. const data = await api.requestPromise(
  79. `/organizations/${organization.slug}/members/`,
  80. {
  81. query: {query},
  82. }
  83. );
  84. this.setState({
  85. orgMembers: data,
  86. dropdownBusy: false,
  87. });
  88. } catch (_err) {
  89. addErrorMessage(t('Unable to load organization members.'), {
  90. duration: 2000,
  91. });
  92. this.setState({
  93. dropdownBusy: false,
  94. });
  95. }
  96. };
  97. getEndpoints(): ReturnType<DeprecatedAsyncView['getEndpoints']> {
  98. const {organization, params} = this.props;
  99. return [
  100. [
  101. 'teamMembers',
  102. `/teams/${organization.slug}/${params.teamId}/members/`,
  103. {},
  104. {paginate: true},
  105. ],
  106. ];
  107. }
  108. addTeamMember = (selection: Item) => {
  109. const {organization, params} = this.props;
  110. const {orgMembers, teamMembers} = this.state;
  111. // Reset members list after adding member to team
  112. this.debouncedFetchMembersRequest('');
  113. joinTeam(
  114. this.props.api,
  115. {
  116. orgId: organization.slug,
  117. teamId: params.teamId,
  118. memberId: selection.value,
  119. },
  120. {
  121. success: () => {
  122. const orgMember = orgMembers.find(member => member.id === selection.value);
  123. if (orgMember === undefined) {
  124. return;
  125. }
  126. this.setState({
  127. error: false,
  128. teamMembers: teamMembers.concat([orgMember as TeamMember]),
  129. });
  130. addSuccessMessage(t('Successfully added member to team.'));
  131. },
  132. error: resp => {
  133. const errorMessage =
  134. (resp && resp.responseJSON && resp.responseJSON.detail) ||
  135. t('Unable to add team member.');
  136. addErrorMessage(errorMessage);
  137. },
  138. }
  139. );
  140. };
  141. removeTeamMember = (member: Member) => {
  142. const {organization, params} = this.props;
  143. const {teamMembers} = this.state;
  144. leaveTeam(
  145. this.props.api,
  146. {
  147. orgId: organization.slug,
  148. teamId: params.teamId,
  149. memberId: member.id,
  150. },
  151. {
  152. success: () => {
  153. this.setState({
  154. teamMembers: teamMembers.filter(m => m.id !== member.id),
  155. });
  156. addSuccessMessage(t('Successfully removed member from team.'));
  157. },
  158. error: () =>
  159. addErrorMessage(
  160. t('There was an error while trying to remove a member from the team.')
  161. ),
  162. }
  163. );
  164. };
  165. updateTeamMemberRole = (member: Member, newRole: string) => {
  166. const {organization} = this.props;
  167. const {teamId} = this.props.params;
  168. const endpoint = `/organizations/${organization.slug}/members/${member.id}/teams/${teamId}/`;
  169. this.props.api.request(endpoint, {
  170. method: 'PUT',
  171. data: {teamRole: newRole},
  172. success: data => {
  173. const teamMembers: any = [...this.state.teamMembers];
  174. const i = teamMembers.findIndex(m => m.id === member.id);
  175. teamMembers[i] = {
  176. ...member,
  177. teamRole: data.teamRole,
  178. };
  179. this.setState({teamMembers});
  180. addSuccessMessage(t('Successfully changed role for team member.'));
  181. },
  182. error: () => {
  183. addErrorMessage(
  184. t('There was an error while trying to change the roles for a team member.')
  185. );
  186. },
  187. });
  188. };
  189. /**
  190. * We perform an API request to support orgs with > 100 members (since that's the max API returns)
  191. *
  192. * @param {Event} e React Event when member filter input changes
  193. */
  194. handleMemberFilterChange = (e: React.ChangeEvent<HTMLInputElement>) => {
  195. this.setState({dropdownBusy: true});
  196. this.debouncedFetchMembersRequest(e.target.value);
  197. };
  198. renderDropdown(isTeamAdmin: boolean) {
  199. const {organization, params, team} = this.props;
  200. const {orgMembers} = this.state;
  201. const existingMembers = new Set(this.state.teamMembers.map(member => member.id));
  202. // members can add other members to a team if the `Open Membership` setting is enabled
  203. // otherwise, `org:write` or `team:admin` permissions are required
  204. const hasOpenMembership = !!organization?.openMembership;
  205. const canAddMembers = hasOpenMembership || isTeamAdmin;
  206. const isDropdownDisabled = team.flags['idp:provisioned'];
  207. const items = (orgMembers || [])
  208. .filter(m => !existingMembers.has(m.id))
  209. .map(m => ({
  210. searchKey: `${m.name} ${m.email}`,
  211. value: m.id,
  212. label: (
  213. <StyledUserListElement>
  214. <StyledAvatar user={m} size={24} className="avatar" />
  215. <StyledNameOrEmail>{m.name || m.email}</StyledNameOrEmail>
  216. </StyledUserListElement>
  217. ),
  218. }));
  219. const menuHeader = (
  220. <StyledMembersLabel>
  221. {t('Members')}
  222. <StyledCreateMemberLink
  223. to=""
  224. onClick={() => openInviteMembersModal({source: 'teams'})}
  225. data-test-id="invite-member"
  226. >
  227. {t('Invite Member')}
  228. </StyledCreateMemberLink>
  229. </StyledMembersLabel>
  230. );
  231. return (
  232. <DropdownAutoComplete
  233. closeOnSelect={false}
  234. items={items}
  235. alignMenu="right"
  236. onSelect={
  237. canAddMembers
  238. ? this.addTeamMember
  239. : selection =>
  240. openTeamAccessRequestModal({
  241. teamId: params.teamId,
  242. orgId: organization.slug,
  243. memberId: selection.value,
  244. })
  245. }
  246. menuHeader={menuHeader}
  247. emptyMessage={t('No members')}
  248. onChange={this.handleMemberFilterChange}
  249. busy={this.state.dropdownBusy}
  250. onClose={() => this.debouncedFetchMembersRequest('')}
  251. disabled={isDropdownDisabled}
  252. data-test-id="add-member-menu"
  253. >
  254. {({isOpen}) => (
  255. <DropdownButton
  256. isOpen={isOpen}
  257. size="xs"
  258. data-test-id="add-member"
  259. disabled={isDropdownDisabled}
  260. >
  261. {t('Add Member')}
  262. </DropdownButton>
  263. )}
  264. </DropdownAutoComplete>
  265. );
  266. }
  267. renderPageTextBlock() {
  268. const {organization, team} = this.props;
  269. const {openMembership} = organization;
  270. const isIdpProvisioned = team.flags['idp:provisioned'];
  271. if (isIdpProvisioned) {
  272. return getButtonHelpText(isIdpProvisioned);
  273. }
  274. return openMembership
  275. ? t(
  276. '"Open Membership" is enabled for the organization. Anyone can add members for this team.'
  277. )
  278. : t(
  279. '"Open Membership" is disabled for the organization. Org Owner/Manager/Admin, or Team Admins can add members for this team.'
  280. );
  281. }
  282. renderMembers(isTeamAdmin: boolean) {
  283. const {config, organization, team} = this.props;
  284. const {access} = organization;
  285. // org:admin is a unique scope that only org owners have
  286. const isOrgOwner = access.includes('org:admin');
  287. const {teamMembers, loading} = this.state;
  288. if (loading) {
  289. return <LoadingIndicator />;
  290. }
  291. if (teamMembers.length) {
  292. return teamMembers.map(member => {
  293. return (
  294. <TeamMembersRow
  295. key={member.id}
  296. hasWriteAccess={isTeamAdmin}
  297. isOrgOwner={isOrgOwner}
  298. organization={organization}
  299. team={team}
  300. member={member}
  301. user={config.user}
  302. removeMember={this.removeTeamMember}
  303. updateMemberRole={this.updateTeamMemberRole}
  304. />
  305. );
  306. });
  307. }
  308. return (
  309. <EmptyMessage icon={<IconUser size="xl" />} size="large">
  310. {t('This team has no members')}
  311. </EmptyMessage>
  312. );
  313. }
  314. render() {
  315. if (this.state.error) {
  316. return <LoadingError onRetry={this.fetchData} />;
  317. }
  318. const {organization, team} = this.props;
  319. const {teamMembersPageLinks} = this.state;
  320. const {openMembership} = organization;
  321. const hasOrgWriteAccess = hasEveryAccess(['org:write'], {organization, team});
  322. const hasTeamAdminAccess = hasEveryAccess(['team:admin'], {organization, team});
  323. const isTeamAdmin = hasOrgWriteAccess || hasTeamAdminAccess;
  324. return (
  325. <Fragment>
  326. <TextBlock>{this.renderPageTextBlock()}</TextBlock>
  327. <PermissionAlert
  328. access={openMembership ? ['org:read'] : ['team:write']}
  329. team={team}
  330. />
  331. <Panel>
  332. <StyledPanelHeader hasButtons>
  333. <div>{t('Members')}</div>
  334. <div>
  335. <TeamRoleColumnLabel />
  336. </div>
  337. <div style={{textTransform: 'none'}}>{this.renderDropdown(isTeamAdmin)}</div>
  338. </StyledPanelHeader>
  339. {this.renderMembers(isTeamAdmin)}
  340. </Panel>
  341. <Pagination pageLinks={teamMembersPageLinks} />
  342. </Fragment>
  343. );
  344. }
  345. }
  346. const StyledUserListElement = styled('div')`
  347. display: grid;
  348. grid-template-columns: max-content 1fr;
  349. gap: ${space(0.5)};
  350. align-items: center;
  351. `;
  352. const StyledNameOrEmail = styled('div')`
  353. font-size: ${p => p.theme.fontSizeSmall};
  354. ${p => p.theme.overflowEllipsis};
  355. `;
  356. const StyledAvatar = styled(props => <UserAvatar {...props} />)`
  357. min-width: 1.75em;
  358. min-height: 1.75em;
  359. width: 1.5em;
  360. height: 1.5em;
  361. `;
  362. const StyledMembersLabel = styled('div')`
  363. display: grid;
  364. grid-template-columns: 1fr max-content;
  365. padding: ${space(1)} 0;
  366. font-size: ${p => p.theme.fontSizeExtraSmall};
  367. text-transform: uppercase;
  368. `;
  369. const StyledCreateMemberLink = styled(Link)`
  370. text-transform: none;
  371. `;
  372. const StyledPanelHeader = styled(PanelHeader)`
  373. ${GRID_TEMPLATE}
  374. `;
  375. export default withConfig(withApi(withOrganization(TeamMembers)));