import {useCallback, useEffect} from 'react'; import styled from '@emotion/styled'; import * as Sentry from '@sentry/react'; import emptyStateImg from 'sentry-images/spot/feedback-empty-state.svg'; import {Button} from 'sentry/components/button'; import ButtonBar from 'sentry/components/buttonBar'; import EmptyStateWarning from 'sentry/components/emptyStateWarning'; import {useFeedbackOnboardingSidebarPanel} from 'sentry/components/feedback/useFeedbackOnboarding'; import OnboardingPanel from 'sentry/components/onboardingPanel'; import {t} from 'sentry/locale'; import {trackAnalytics} from 'sentry/utils/analytics'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import useProjects from 'sentry/utils/useProjects'; import useRouter from 'sentry/utils/useRouter'; type Props = { issueTab?: boolean; projectIds?: string[]; }; export function UserFeedbackEmpty({projectIds, issueTab = false}: Props) { const {projects, initiallyLoaded} = useProjects(); const loadingProjects = !initiallyLoaded; const organization = useOrganization(); const location = useLocation(); const selectedProjects = projectIds?.length ? projects.filter(({id}) => projectIds.includes(id)) : projects; const hasAnyFeedback = selectedProjects.some(({hasUserReports}) => hasUserReports); const hasNewOnboarding = organization.features.includes('user-feedback-onboarding'); const {activateSidebarIssueDetails} = useFeedbackOnboardingSidebarPanel(); const router = useRouter(); const setProjId = useCallback(() => { router.push({ pathname: location.pathname, query: {...location.query, project: projectIds?.[0]}, hash: location.hash, }); }, [location.hash, location.query, location.pathname, projectIds, router]); useEffect(() => { if (issueTab) { setProjId(); } // eslint-disable-next-line react-hooks/exhaustive-deps }, []); useEffect(() => { 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; setTimeout(() => { this._submitInProgress = false; this.onSuccess(); }, 500); }; }; if (hasAnyFeedback === false) { // send to reload only due to higher event volume trackAnalytics('user_feedback.viewed', { organization, projects: projectIds?.join(',') || '', }); } return () => { window.sentryEmbedCallback = null; }; }, [hasAnyFeedback, organization, projectIds]); function trackAnalyticsInternal( eventKey: 'user_feedback.docs_clicked' | 'user_feedback.dialog_opened' ) { trackAnalytics(eventKey, { organization, projects: selectedProjects?.join(','), }); } // Show no user reports if waiting for projects to load or if there is no feedback if (loadingProjects || hasAnyFeedback !== false) { return (

{t('Sorry, no user reports match your filters.')}

); } // Show landing page after projects have loaded and it is confirmed no projects have feedback return ( } >

{t('What do users think?')}

{t( `You can't read minds. At least we hope not. Ask users for feedback on the impact of their crashes or bugs and you shall receive.` )}

{hasNewOnboarding ? ( ) : ( )}
); } const ButtonList = styled(ButtonBar)` grid-template-columns: repeat(auto-fit, minmax(130px, max-content)); `;