import {useRef, useState} from 'react'; import styled from '@emotion/styled'; import FeatureBadge from 'sentry/components/badge/featureBadge'; import {Button} from 'sentry/components/button'; import {Chevron} from 'sentry/components/chevron'; import useDrawer from 'sentry/components/globalDrawer'; import {GroupSummary, useGroupSummary} from 'sentry/components/group/groupSummary'; import Placeholder from 'sentry/components/placeholder'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {Event} from 'sentry/types/event'; import type {Group} from 'sentry/types/group'; import type {Project} from 'sentry/types/project'; import {getConfigForIssueType} from 'sentry/utils/issueTypeConfig'; import {singleLineRenderer} from 'sentry/utils/marked'; import {useAiConfig} from 'sentry/views/issueDetails/streamline/hooks/useAiConfig'; import Resources from 'sentry/views/issueDetails/streamline/sidebar/resources'; import {SidebarSectionTitle} from 'sentry/views/issueDetails/streamline/sidebar/sidebar'; import {SolutionsHubDrawer} from 'sentry/views/issueDetails/streamline/sidebar/solutionsHubDrawer'; import {useHasStreamlinedUI} from 'sentry/views/issueDetails/utils'; export default function SolutionsSection({ group, project, event, }: { event: Event | undefined; group: Group; project: Project; }) { const [isExpanded, setIsExpanded] = useState(false); const openButtonRef = useRef(null); const {openDrawer} = useDrawer(); const hasStreamlinedUI = useHasStreamlinedUI(); const openSolutionsDrawer = () => { if (!event) { return; } openDrawer( () => , { ariaLabel: t('Solutions drawer'), // We prevent a click on the Open/Close Autofix button from closing the drawer so that // we don't reopen it immediately, and instead let the button handle this itself. shouldCloseOnInteractOutside: element => { const viewAllButton = openButtonRef.current; if ( viewAllButton?.contains(element) || document.getElementById('sentry-feedback')?.contains(element) || document.getElementById('autofix-rethink-input')?.contains(element) || document.getElementById('autofix-write-access-modal')?.contains(element) ) { return false; } return true; }, transitionProps: {stiffness: 1000}, } ); }; const { data: summaryData, isPending: isSummaryLoading, isError: isSummaryError, } = useGroupSummary(group, event, project); const aiConfig = useAiConfig(group, event, project); const issueTypeConfig = getConfigForIssueType(group, project); const showCtaButton = aiConfig.needsGenAIConsent || aiConfig.hasAutofix || (aiConfig.hasSummary && aiConfig.hasResources); const isButtonLoading = aiConfig.isAutofixSetupLoading; const getButtonText = () => { if (aiConfig.needsGenAIConsent) { return t('Set up Sentry AI'); } if (aiConfig.hasAutofix) { if (aiConfig.needsAutofixSetup) { return t('Set up Autofix'); } return aiConfig.hasResources ? t('Open Resources & Autofix') : t('Open Autofix'); } return t('Open Resources'); }; const renderContent = () => { if (aiConfig.needsGenAIConsent) { return ( ); } // Show the summary's loading state if we're still loading the autofix setup if (aiConfig.hasSummary) { return ( ); } if (!aiConfig.hasSummary && issueTypeConfig.resources) { return ( setIsExpanded(!isExpanded)} size="zero"> {isExpanded ? t('SHOW LESS') : t('READ MORE')} ); } return null; }; return ( {t('Solutions Hub')} {aiConfig.hasSummary && ( , } )} /> )} {renderContent()} {isButtonLoading ? ( ) : showCtaButton ? ( openSolutionsDrawer()} analyticsEventKey="issue_details.solutions_hub_opened" analyticsEventName="Issue Details: Solutions Hub Opened" analyticsParams={{ has_streamlined_ui: hasStreamlinedUI, }} > {getButtonText()} ) : null} ); } const SolutionsSectionContainer = styled('div')` display: flex; flex-direction: column; `; const Summary = styled('div')` margin-bottom: ${space(0.5)}; position: relative; `; const HeadlineText = styled('span')` margin-right: ${space(0.5)}; word-break: break-word; `; const ResourcesWrapper = styled('div')<{isExpanded: boolean}>` position: relative; margin-bottom: ${space(1)}; `; const ResourcesContent = styled('div')<{isExpanded: boolean}>` position: relative; max-height: ${p => (p.isExpanded ? 'none' : '68px')}; overflow: hidden; padding-bottom: ${p => (p.isExpanded ? space(2) : 0)}; ${p => !p.isExpanded && ` &::after { content: ''; position: absolute; bottom: 0; left: 0; right: 0; height: 40px; background: linear-gradient(transparent, ${p.theme.background}); } `} `; const ExpandButton = styled(Button)` position: absolute; bottom: -${space(1)}; right: 0; font-size: ${p => p.theme.fontSizeExtraSmall}; color: ${p => p.theme.gray300}; border: none; box-shadow: none; &:hover { color: ${p => p.theme.gray400}; } `; const StyledButton = styled(Button)` margin-top: ${space(1)}; width: 100%; background: ${p => p.theme.background} linear-gradient(to right, ${p => p.theme.background}, ${p => p.theme.pink400}20); color: ${p => p.theme.pink400}; `; const ChevronContainer = styled('div')` margin-left: ${space(0.5)}; height: 16px; width: 16px; `; const HeaderContainer = styled('div')` display: flex; align-items: center; gap: ${space(0.5)}; `; const StyledFeatureBadge = styled(FeatureBadge)` margin-left: ${space(0.25)}; padding-bottom: 3px; `; const ButtonPlaceholder = styled(Placeholder)` width: 100%; height: 38px; border-radius: ${p => p.theme.borderRadius}; margin-top: ${space(1)}; `;