import {Fragment} from 'react'; import AlertLink from 'sentry/components/alertLink'; import AsyncComponent from 'sentry/components/asyncComponent'; import Form from 'sentry/components/forms/form'; import JsonForm from 'sentry/components/forms/jsonForm'; import FormModel from 'sentry/components/forms/model'; import {FieldObject} from 'sentry/components/forms/types'; import Link from 'sentry/components/links/link'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import {IconMail} from 'sentry/icons'; import {t} from 'sentry/locale'; import {Organization} from 'sentry/types'; import {trackAnalytics} from 'sentry/utils/analytics'; import withOrganizations from 'sentry/utils/withOrganizations'; import { CONFIRMATION_MESSAGE, NOTIFICATION_FEATURE_MAP, NOTIFICATION_SETTINGS_PATHNAMES, NOTIFICATION_SETTINGS_TYPES, NotificationSettingsObject, SELF_NOTIFICATION_SETTINGS_TYPES, } from 'sentry/views/settings/account/notifications/constants'; import {NOTIFICATION_SETTING_FIELDS} from 'sentry/views/settings/account/notifications/fields2'; import { decideDefault, getParentIds, getStateToPutForDefault, isSufficientlyComplex, mergeNotificationSettings, } from 'sentry/views/settings/account/notifications/utils'; import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader'; import TextBlock from 'sentry/views/settings/components/text/textBlock'; type Props = AsyncComponent['props'] & { organizations: Organization[]; }; type State = { legacyData: {[key: string]: string}; notificationSettings: NotificationSettingsObject; } & AsyncComponent['state']; class NotificationSettings extends AsyncComponent { model = new FormModel(); getDefaultState(): State { return { ...super.getDefaultState(), notificationSettings: {}, legacyData: {}, }; } getEndpoints(): ReturnType { return [ ['notificationSettings', `/users/me/notification-settings/`], ['legacyData', '/users/me/notifications/'], ]; } componentDidMount() { super.componentDidMount(); // only tied to a user trackAnalytics('notification_settings.index_page_viewed', { organization: null, }); } getStateToPutForDefault = ( changedData: {[key: string]: string}, notificationType: string ) => { /** * Update the current providers' parent-independent notification settings * with the new value. If the new value is "never", then also update all * parent-specific notification settings to "default". If the previous value * was "never", then assume providerList should be "email" only. */ const {notificationSettings} = this.state; const updatedNotificationSettings = getStateToPutForDefault( notificationType, notificationSettings, changedData, getParentIds(notificationType, notificationSettings) ); this.setState({ notificationSettings: mergeNotificationSettings( notificationSettings, updatedNotificationSettings ), }); return updatedNotificationSettings; }; get notificationSettingsType() { // filter out notification settings if the feature flag isn't set return NOTIFICATION_SETTINGS_TYPES.filter(type => { const notificationFlag = NOTIFICATION_FEATURE_MAP[type]; if (notificationFlag) { return this.props.organizations.some(org => org.features?.includes(notificationFlag) ); } return true; }); } getInitialData(): {[key: string]: string} { const {notificationSettings, legacyData} = this.state; const notificationsInitialData = Object.fromEntries( this.notificationSettingsType.map(notificationType => [ notificationType, decideDefault(notificationType, notificationSettings), ]) ); const allInitialData = { ...notificationsInitialData, ...legacyData, }; return allInitialData; } getFields(): FieldObject[] { const {notificationSettings} = this.state; const fields: FieldObject[] = []; const endOfFields: FieldObject[] = []; for (const notificationType of this.notificationSettingsType) { const field = Object.assign({}, NOTIFICATION_SETTING_FIELDS[notificationType], { getData: data => this.getStateToPutForDefault(data, notificationType), help: (

{NOTIFICATION_SETTING_FIELDS[notificationType].help}   Fine tune

), }) as any; if ( isSufficientlyComplex(notificationType, notificationSettings) && typeof field !== 'function' ) { field.confirm = {never: CONFIRMATION_MESSAGE}; } if (field.type === 'blank') { endOfFields.push(field); } else { fields.push(field); } } const legacyField = SELF_NOTIFICATION_SETTINGS_TYPES.map( type => NOTIFICATION_SETTING_FIELDS[type] as FieldObject ); fields.push(...legacyField); const allFields = [...fields, ...endOfFields]; return allFields; } onFieldChange = (fieldName: string) => { if (SELF_NOTIFICATION_SETTINGS_TYPES.includes(fieldName)) { this.model.setFormOptions({apiEndpoint: '/users/me/notifications/'}); } else { this.model.setFormOptions({apiEndpoint: '/users/me/notification-settings/'}); } }; renderBody() { return ( {t('Personal notifications sent by email or an integration.')}
}> {t('Looking to add or remove an email address? Use the emails panel.')}
); } } export default withOrganizations(NotificationSettings);