ruvector/studio/components/layouts/ProjectLayout/LayoutHeader/HelpPopover.tsx
rUv 814f595995 feat(studio): Add complete RuVector Studio application
Major additions:
- Complete Next.js studio application with 1600+ components
- Docker support (Dockerfile.combined, docker-compose.yml)
- GCP deployment documentation and benchmarks
- SQL benchmark scripts for performance testing
- Sentry integration for monitoring
- Comprehensive test suite and mocks

Studio features:
- Dashboard and admin interfaces
- Data visualization components
- Authentication and user management
- API integration with RuVector backend
- Static data and public assets

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-06 23:04:48 +00:00

200 lines
7.8 KiB
TypeScript

import { Activity, BookOpen, HelpCircle, Mail, Wrench } from 'lucide-react'
import Image from 'next/legacy/image'
import { useRouter } from 'next/router'
import { useState } from 'react'
import SVG from 'react-inlinesvg'
import { IS_PLATFORM } from 'common'
import type { SupportFormUrlKeys } from 'components/interfaces/Support/SupportForm.utils'
import { SupportLink } from 'components/interfaces/Support/SupportLink'
import { SIDEBAR_KEYS } from 'components/layouts/ProjectLayout/LayoutSidebar/LayoutSidebarProvider'
import { ButtonTooltip } from 'components/ui/ButtonTooltip'
import { useSendEventMutation } from 'data/telemetry/send-event-mutation'
import { useSelectedOrganizationQuery } from 'hooks/misc/useSelectedOrganization'
import { useSelectedProjectQuery } from 'hooks/misc/useSelectedProject'
import { DOCS_URL } from 'lib/constants'
import { useAiAssistantStateSnapshot } from 'state/ai-assistant-state'
import { useSidebarManagerSnapshot } from 'state/sidebar-manager-state'
import {
AiIconAnimation,
Button,
ButtonGroup,
ButtonGroupItem,
cn,
Popover,
Popover_Shadcn_,
PopoverContent_Shadcn_,
PopoverTrigger_Shadcn_,
} from 'ui'
export const HelpPopover = () => {
const router = useRouter()
const { data: project } = useSelectedProjectQuery()
const { data: org } = useSelectedOrganizationQuery()
const snap = useAiAssistantStateSnapshot()
const { openSidebar } = useSidebarManagerSnapshot()
const { mutate: sendEvent } = useSendEventMutation()
const [isOpen, setIsOpen] = useState(false)
const projectRef = project?.parent_project_ref ?? (router.query.ref as string | undefined)
let supportLinkQueryParams: Partial<SupportFormUrlKeys> | undefined = undefined
if (projectRef) {
supportLinkQueryParams = { projectRef }
} else if (org?.slug) {
supportLinkQueryParams = { orgSlug: org.slug }
}
return (
<Popover_Shadcn_ open={isOpen} onOpenChange={setIsOpen}>
<PopoverTrigger_Shadcn_ asChild>
<ButtonTooltip
id="help-popover-button"
type="outline"
size="tiny"
className={cn(
'rounded-full w-[32px] h-[32px] flex items-center justify-center p-0 group',
isOpen && 'bg-foreground text-background'
)}
onClick={() => {
sendEvent({
action: 'help_button_clicked',
groups: { project: project?.ref, organization: org?.slug },
})
}}
tooltip={{ content: { side: 'bottom', text: 'Help' } }}
>
<HelpCircle
size={16}
strokeWidth={1.5}
className={cn(
'text-foreground-light group-hover:text-foreground',
isOpen && 'text-background group-hover:text-background'
)}
/>
</ButtonTooltip>
</PopoverTrigger_Shadcn_>
<PopoverContent_Shadcn_ className="w-[400px] space-y-5 p-0 py-5" align="end" side="bottom">
<div className="flex flex-col gap-4">
<div className="px-5 flex flex-col gap-1">
<h5 className="text-foreground">Need help with your project?</h5>
<p className="text-sm text-foreground-lighter text-balance">
Start with our {projectRef ? 'Assistant, docs,' : 'docs'} or community.
</p>
</div>
<div className="px-5">
<ButtonGroup className="w-full">
{projectRef && (
<ButtonGroupItem
size="tiny"
icon={<AiIconAnimation allowHoverEffect size={14} />}
onClick={() => {
openSidebar(SIDEBAR_KEYS.AI_ASSISTANT)
snap.newChat({
name: 'Support',
initialInput: `I need help with my project`,
suggestions: {
title:
'I can help you with your project, here are some example prompts to get you started:',
prompts: [
{
label: 'Database Health',
description: 'Summarise my database health and performance',
},
{
label: 'Debug Logs',
description: 'View and debug my edge function logs',
},
{
label: 'RLS Setup',
description: 'Implement row level security for my tables',
},
],
},
})
}}
>
Supabase Assistant
</ButtonGroupItem>
)}
<ButtonGroupItem size="tiny" icon={<BookOpen strokeWidth={1.5} size={14} />} asChild>
<a href={`${DOCS_URL}/`} target="_blank" rel="noreferrer">
Docs
</a>
</ButtonGroupItem>
<ButtonGroupItem size="tiny" icon={<Wrench strokeWidth={1.5} size={14} />} asChild>
<a
href={`${DOCS_URL}/guides/troubleshooting?products=platform`}
target="_blank"
rel="noreferrer"
>
Troubleshooting
</a>
</ButtonGroupItem>
{IS_PLATFORM && (
<>
<ButtonGroupItem
asChild
size="tiny"
icon={<Activity strokeWidth={1.5} size={14} />}
>
<a href="https://status.supabase.com/" target="_blank" rel="noreferrer">
Supabase status
</a>
</ButtonGroupItem>
<ButtonGroupItem asChild size="tiny" icon={<Mail strokeWidth={1.5} size={14} />}>
<SupportLink
queryParams={supportLinkQueryParams}
onClick={() => setIsOpen(false)}
>
Contact support
</SupportLink>
</ButtonGroupItem>
</>
)}
</ButtonGroup>
</div>
</div>
<Popover.Separator />
<div className="flex flex-col gap-4">
<div className="px-5 flex flex-col gap-1">
<h5 className="text-foreground">Community support</h5>
<p className="text-sm text-foreground-lighter text-balance">
Our Discord community can help with code-related issues. Many questions are answered
in minutes.
</p>
</div>
<div className="px-5">
<div
className="relative space-y-2 overflow-hidden rounded px-5 py-4 pb-12 shadow-md"
style={{ background: '#404EED' }}
>
<a
href="https://discord.supabase.com"
target="_blank"
rel="noreferrer"
className="group dark block cursor-pointer"
>
<Image
className="absolute left-0 top-0 opacity-50 transition-opacity group-hover:opacity-40"
src={`${router.basePath}/img/support/discord-bg-small.jpg`}
layout="fill"
objectFit="cover"
alt="Discord illustration"
/>
<Button
type="secondary"
size="tiny"
icon={<SVG src={`${router.basePath}/img/discord-icon.svg`} className="h-4 w-4" />}
>
<span style={{ color: '#404EED' }}>Join us on Discord</span>
</Button>
</a>
</div>
</div>
</div>
</PopoverContent_Shadcn_>
</Popover_Shadcn_>
)
}