import { AnimatePresence, motion } from 'framer-motion' import { ChevronLeft } from 'lucide-react' import Link from 'next/link' import { ReactNode, useMemo } from 'react' import { LOCAL_STORAGE_KEYS, useParams } from 'common' import { useIsBranching2Enabled } from 'components/interfaces/App/FeaturePreview/FeaturePreviewContext' import { Connect } from 'components/interfaces/Connect/Connect' import { LocalDropdown } from 'components/interfaces/LocalDropdown' import { UserDropdown } from 'components/interfaces/UserDropdown' import { AdvisorButton } from 'components/layouts/AppLayout/AdvisorButton' import { AssistantButton } from 'components/layouts/AppLayout/AssistantButton' import { BranchDropdown } from 'components/layouts/AppLayout/BranchDropdown' import { InlineEditorButton } from 'components/layouts/AppLayout/InlineEditorButton' import { OrganizationDropdown } from 'components/layouts/AppLayout/OrganizationDropdown' import { ProjectDropdown } from 'components/layouts/AppLayout/ProjectDropdown' import { getResourcesExceededLimitsOrg } from 'components/ui/OveragesBanner/OveragesBanner.utils' import { useOrgUsageQuery } from 'data/usage/org-usage-query' import { useLocalStorageQuery } from 'hooks/misc/useLocalStorage' import { useSelectedOrganizationQuery } from 'hooks/misc/useSelectedOrganization' import { useSelectedProjectQuery } from 'hooks/misc/useSelectedProject' import { IS_PLATFORM } from 'lib/constants' import { useRouter } from 'next/router' import { useAppStateSnapshot } from 'state/app-state' import { Badge, cn } from 'ui' import { CommandMenuTriggerInput } from 'ui-patterns' import { BreadcrumbsView } from './BreadcrumbsView' import { FeedbackDropdown } from './FeedbackDropdown/FeedbackDropdown' import { HelpPopover } from './HelpPopover' import { HomeIcon } from './HomeIcon' import { LocalVersionPopover } from './LocalVersionPopover' import MergeRequestButton from './MergeRequestButton' const LayoutHeaderDivider = ({ className, ...props }: React.HTMLProps) => ( ) interface LayoutHeaderProps { customHeaderComponents?: ReactNode breadcrumbs?: any[] headerTitle?: string showProductMenu?: boolean backToDashboardURL?: string } export const LayoutHeader = ({ customHeaderComponents, breadcrumbs = [], headerTitle, showProductMenu, backToDashboardURL, }: LayoutHeaderProps) => { const { ref: projectRef, slug } = useParams() const router = useRouter() const { data: selectedProject } = useSelectedProjectQuery() const { data: selectedOrganization } = useSelectedOrganizationQuery() const { setMobileMenuOpen } = useAppStateSnapshot() const gitlessBranching = useIsBranching2Enabled() const [commandMenuEnabled] = useLocalStorageQuery(LOCAL_STORAGE_KEYS.HOTKEY_COMMAND_MENU, true) const isAccountPage = router.pathname.startsWith('/account') // We only want to query the org usage and check for possible over-ages for plans without usage billing enabled (free or pro with spend cap) const { data: orgUsage } = useOrgUsageQuery( { orgSlug: selectedOrganization?.slug }, { enabled: selectedOrganization?.usage_billing_enabled === false } ) const exceedingLimits = useMemo(() => { if (orgUsage) { return getResourcesExceededLimitsOrg(orgUsage?.usages || []).length > 0 } else { return false } }, [orgUsage]) // show org selection if we are on a project page or on a explicit org route const showOrgSelection = slug || (selectedOrganization && projectRef) return ( <>
{backToDashboardURL && isAccountPage && (
)} {(showProductMenu || isAccountPage) && (
)}
{showOrgSelection && IS_PLATFORM ? ( <> ) : null} {projectRef && ( {exceedingLimits && (
Exceeding usage limits
)} {selectedProject && ( <> {IS_PLATFORM && } )}
)}
{headerTitle && ( {headerTitle} )}
{projectRef && ( {IS_PLATFORM && gitlessBranching && } )}
{customHeaderComponents && customHeaderComponents} {IS_PLATFORM ? ( <>
div]:border-none', '[&_.command-shortcut>div]:pr-2', '[&_.command-shortcut>div]:bg-transparent', '[&_.command-shortcut>div]:text-foreground-lighter' )} /> {!!projectRef && ( <> )}
) : ( <>
{!!projectRef && ( <> )}
)}
) }