index.tsx 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. import {Fragment} from 'react';
  2. import {browserHistory, RouteComponentProps} from 'react-router';
  3. import {addLoadingMessage} from 'sentry/actionCreators/indicator';
  4. import {
  5. changeOrganizationSlug,
  6. removeAndRedirectToRemainingOrganization,
  7. updateOrganization,
  8. } from 'sentry/actionCreators/organizations';
  9. import {Button} from 'sentry/components/button';
  10. import Confirm from 'sentry/components/confirm';
  11. import FieldGroup from 'sentry/components/forms/fieldGroup';
  12. import List from 'sentry/components/list';
  13. import ListItem from 'sentry/components/list/listItem';
  14. import Panel from 'sentry/components/panels/panel';
  15. import PanelHeader from 'sentry/components/panels/panelHeader';
  16. import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
  17. import {t, tct} from 'sentry/locale';
  18. import {Organization, Project} from 'sentry/types';
  19. import {trackAnalytics} from 'sentry/utils/analytics';
  20. import useApi from 'sentry/utils/useApi';
  21. import useOrganization from 'sentry/utils/useOrganization';
  22. import withProjects from 'sentry/utils/withProjects';
  23. import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
  24. import TextBlock from 'sentry/views/settings/components/text/textBlock';
  25. import PermissionAlert from 'sentry/views/settings/organization/permissionAlert';
  26. import {OrganizationRegionAction} from 'sentry/views/settings/organizationGeneralSettings/organizationRegionAction';
  27. import OrganizationSettingsForm from './organizationSettingsForm';
  28. type Props = {
  29. projects: Project[];
  30. } & RouteComponentProps<{}, {}>;
  31. function OrganizationGeneralSettings({projects}: Props) {
  32. const api = useApi();
  33. const organization = useOrganization();
  34. const removeConfirmMessage = (
  35. <Fragment>
  36. <TextBlock>
  37. {tct(
  38. 'Removing the organization, [name] is permanent and cannot be undone! Are you sure you want to continue?',
  39. {
  40. name: organization && <strong>{organization.name}</strong>,
  41. }
  42. )}
  43. </TextBlock>
  44. {!!projects.length && (
  45. <Fragment>
  46. <TextBlock>
  47. {t('This will also remove the following associated projects:')}
  48. </TextBlock>
  49. <List symbol="bullet" data-test-id="removed-projects-list">
  50. {projects.map(project => (
  51. <ListItem key={project.slug}>{project.slug}</ListItem>
  52. ))}
  53. </List>
  54. </Fragment>
  55. )}
  56. </Fragment>
  57. );
  58. const handleSaveForm: React.ComponentProps<
  59. typeof OrganizationSettingsForm
  60. >['onSave'] = (prevData: Organization, updated: Organization) => {
  61. if (updated.slug && updated.slug !== prevData.slug) {
  62. changeOrganizationSlug(prevData, updated);
  63. if (updated.features.includes('customer-domains')) {
  64. const {organizationUrl} = updated.links;
  65. window.location.replace(`${organizationUrl}/settings/organization/`);
  66. } else {
  67. browserHistory.replace(`/settings/${updated.slug}/`);
  68. }
  69. } else {
  70. if (prevData.codecovAccess !== updated.codecovAccess) {
  71. trackAnalytics('organization_settings.codecov_access_updated', {
  72. organization: updated,
  73. has_access: updated.codecovAccess,
  74. });
  75. }
  76. // This will update OrganizationStore (as well as OrganizationsStore
  77. // which is slightly incorrect because it has summaries vs a detailed org)
  78. updateOrganization(updated);
  79. }
  80. };
  81. const handleConfirmRemoveOrg = () => {
  82. if (!organization) {
  83. return;
  84. }
  85. addLoadingMessage();
  86. removeAndRedirectToRemainingOrganization(api, {
  87. orgId: organization.slug,
  88. successMessage: `${organization.name} is queued for deletion.`,
  89. errorMessage: `Error removing the ${organization.name} organization`,
  90. });
  91. };
  92. const organizationRegionInfo = OrganizationRegionAction({
  93. organization,
  94. });
  95. return (
  96. <Fragment>
  97. <SentryDocumentTitle title={t('General Settings')} orgSlug={organization.slug} />
  98. <div>
  99. <SettingsPageHeader
  100. title={t('Organization Settings')}
  101. action={organizationRegionInfo}
  102. />
  103. <PermissionAlert />
  104. <OrganizationSettingsForm initialData={organization} onSave={handleSaveForm} />
  105. {organization.access.includes('org:admin') && !organization.isDefault && (
  106. <Panel>
  107. <PanelHeader>{t('Remove Organization')}</PanelHeader>
  108. <FieldGroup
  109. label={t('Remove Organization')}
  110. help={t(
  111. 'Removing this organization will delete all data including projects and their associated events.'
  112. )}
  113. >
  114. <div>
  115. <Confirm
  116. priority="danger"
  117. confirmText={t('Remove Organization')}
  118. message={removeConfirmMessage}
  119. onConfirm={handleConfirmRemoveOrg}
  120. >
  121. <Button priority="danger">{t('Remove Organization')}</Button>
  122. </Confirm>
  123. </div>
  124. </FieldGroup>
  125. </Panel>
  126. )}
  127. </div>
  128. </Fragment>
  129. );
  130. }
  131. export default withProjects(OrganizationGeneralSettings);