import type {RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import * as Sentry from '@sentry/react'; import Access from 'sentry/components/acl/access'; import {Button} from 'sentry/components/button'; import Form from 'sentry/components/forms/form'; import JsonForm from 'sentry/components/forms/jsonForm'; import formGroups from 'sentry/data/forms/userFeedback'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Organization} from 'sentry/types/organization'; import type {Project} from 'sentry/types/project'; import routeTitleGen from 'sentry/utils/routeTitle'; import withOrganization from 'sentry/utils/withOrganization'; import DeprecatedAsyncView from 'sentry/views/deprecatedAsyncView'; import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader'; import TextBlock from 'sentry/views/settings/components/text/textBlock'; import PermissionAlert from 'sentry/views/settings/project/permissionAlert'; type RouteParams = { projectId: string; }; type Props = RouteComponentProps & { organization: Organization; project: Project; }; class ProjectUserFeedbackSettings extends DeprecatedAsyncView { submitTimeout: number | undefined = undefined; componentDidMount() { super.componentDidMount(); window.sentryEmbedCallback = function (embed) { // Mock the embed's submit xhr to always be successful // NOTE: this will not have errors if the form is empty embed.submit = function (_body) { this._submitInProgress = true; window.setTimeout(() => { this._submitInProgress = false; this.onSuccess(); }, 500); }; }; } componentWillUnmount() { window.sentryEmbedCallback = null; } getEndpoints(): ReturnType { const {organization} = this.props; const {projectId} = this.props.params; return [ ['keyList', `/projects/${organization.slug}/${projectId}/keys/`], ['project', `/projects/${organization.slug}/${projectId}/`], ]; } getTitle(): string { const {projectId} = this.props.params; return routeTitleGen(t('User Feedback'), projectId, false); } handleClick = () => { Sentry.showReportDialog({ // should never make it to the Sentry API, but just in case, use throwaway id eventId: '00000000000000000000000000000000', }); }; renderBody() { const {organization, project} = this.props; const {projectId} = this.props.params; return (
} /> {t( `Don't rely on stack traces and graphs alone to understand the cause and impact of errors. Enable User Feedback to collect your users' comments when they encounter a crash or bug.` )} {t( `When configured, your users will be presented with a dialog prompting them for additional information. That information will get attached to the issue in Sentry.` )}
{({hasAccess}) => }
); } } const ButtonList = styled('div')` display: inline-grid; grid-auto-flow: column; gap: ${space(1)}; `; export default withOrganization(ProjectUserFeedbackSettings);