import {Fragment} from 'react'; import styled from '@emotion/styled'; import {Breadcrumbs, type Crumb} from 'sentry/components/breadcrumbs'; import ButtonBar from 'sentry/components/buttonBar'; import FeedbackWidgetButton from 'sentry/components/feedback/widget/feedbackWidgetButton'; import * as Layout from 'sentry/components/layouts/thirds'; import {TabList, Tabs} from 'sentry/components/tabs'; import {IconBusiness} from 'sentry/icons'; import {space} from 'sentry/styles/space'; import {useNavigate} from 'sentry/utils/useNavigate'; import useOrganization from 'sentry/utils/useOrganization'; import {useModuleTitles} from 'sentry/views/insights/common/utils/useModuleTitle'; import { type RoutableModuleNames, useModuleURLBuilder, } from 'sentry/views/insights/common/utils/useModuleURL'; import { DOMAIN_VIEW_BASE_TITLE, OVERVIEW_PAGE_TITLE, } from 'sentry/views/insights/pages/settings'; import {isModuleEnabled} from 'sentry/views/insights/pages/utils'; import type {ModuleName} from 'sentry/views/insights/types'; export type Props = { domainBaseUrl: string; domainTitle: string; headerTitle: React.ReactNode; modules: ModuleName[]; selectedModule: ModuleName | undefined; additionalBreadCrumbs?: Crumb[]; additonalHeaderActions?: React.ReactNode; hideDefaultTabs?: boolean; tabs?: {onTabChange: (key: string) => void; tabList: React.ReactNode; value: string}; }; type Tab = { key: string; label: React.ReactNode; }; export function DomainViewHeader({ modules, headerTitle, domainTitle, selectedModule, hideDefaultTabs, additonalHeaderActions, additionalBreadCrumbs = [], domainBaseUrl, tabs, }: Props) { const navigate = useNavigate(); const organization = useOrganization(); const moduleURLBuilder = useModuleURLBuilder(); const moduleTitles = useModuleTitles(); const baseCrumbs: Crumb[] = [ { label: DOMAIN_VIEW_BASE_TITLE, to: undefined, // There is no base /performance/ page preservePageFilters: true, }, { label: domainTitle, to: domainBaseUrl, preservePageFilters: true, }, { label: selectedModule ? moduleTitles[selectedModule] : OVERVIEW_PAGE_TITLE, to: selectedModule ? `${moduleURLBuilder(selectedModule as RoutableModuleNames)}/` : domainBaseUrl, preservePageFilters: true, }, ...additionalBreadCrumbs, ]; const showModuleTabs = organization.features.includes('insights-entry-points'); const defaultHandleTabChange = (key: ModuleName | typeof OVERVIEW_PAGE_TITLE) => { if (key === selectedModule || (key === OVERVIEW_PAGE_TITLE && !module)) { return; } if (!key) { return; } if (key === OVERVIEW_PAGE_TITLE) { navigate(domainBaseUrl); return; } navigate(`${moduleURLBuilder(key as RoutableModuleNames)}/`); }; const tabValue = hideDefaultTabs && tabs?.value ? tabs.value : selectedModule ?? OVERVIEW_PAGE_TITLE; const handleTabChange = hideDefaultTabs && tabs ? tabs.onTabChange : defaultHandleTabChange; const tabList: Tab[] = [ { key: OVERVIEW_PAGE_TITLE, label: OVERVIEW_PAGE_TITLE, }, ]; if (showModuleTabs) { tabList.push( ...modules.map(moduleName => ({ key: moduleName, label: , })) ); } return ( {headerTitle} {additonalHeaderActions} {!hideDefaultTabs && ( {tabList.map(tab => ( {tab.label} ))} )} {hideDefaultTabs && tabs && tabs.tabList} ); } function TabLabel({moduleName}: {moduleName: ModuleName}) { const moduleTitles = useModuleTitles(); const organization = useOrganization(); const showBusinessIcon = !isModuleEnabled(moduleName, organization); if (showBusinessIcon) { return ( {moduleTitles[moduleName]} ); } return {moduleTitles[moduleName]}; } const TabWithIconContainer = styled('div')` display: inline-flex; align-items: center; text-align: left; gap: ${space(0.5)}; `;