mirror of
https://github.com/ruvnet/RuVector.git
synced 2026-05-25 15:03:46 +00:00
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>
331 lines
12 KiB
TypeScript
331 lines
12 KiB
TypeScript
import { Code, Table2 } from 'lucide-react'
|
|
import { useRouter } from 'next/router'
|
|
import { useCallback, useMemo, useState } from 'react'
|
|
|
|
import { useParams } from 'common'
|
|
import { FRAMEWORKS } from 'components/interfaces/Connect/Connect.constants'
|
|
import { SIDEBAR_KEYS } from 'components/layouts/ProjectLayout/LayoutSidebar/LayoutSidebarProvider'
|
|
import { useSendEventMutation } from 'data/telemetry/send-event-mutation'
|
|
import { useSelectedOrganizationQuery } from 'hooks/misc/useSelectedOrganization'
|
|
import { useSelectedProjectQuery } from 'hooks/misc/useSelectedProject'
|
|
import { BASE_PATH } from 'lib/constants'
|
|
import { useAiAssistantStateSnapshot } from 'state/ai-assistant-state'
|
|
import { useSidebarManagerSnapshot } from 'state/sidebar-manager-state'
|
|
import { Button, Card, CardContent, ToggleGroup, ToggleGroupItem } from 'ui'
|
|
import { FrameworkSelector } from './FrameworkSelector'
|
|
import { GettingStarted } from './GettingStarted'
|
|
import {
|
|
GettingStartedAction,
|
|
GettingStartedState,
|
|
GettingStartedStep,
|
|
} from './GettingStarted.types'
|
|
import {
|
|
DEFAULT_FRAMEWORK_KEY,
|
|
getCodeWorkflowSteps,
|
|
getNoCodeWorkflowSteps,
|
|
} from './GettingStarted.utils'
|
|
import { useGettingStartedProgress } from './useGettingStartedProgress'
|
|
|
|
interface GettingStartedSectionProps {
|
|
value: GettingStartedState
|
|
onChange: (v: GettingStartedState) => void
|
|
}
|
|
|
|
export function GettingStartedSection({ value, onChange }: GettingStartedSectionProps) {
|
|
const router = useRouter()
|
|
const { ref } = useParams()
|
|
const { data: project } = useSelectedProjectQuery()
|
|
const { data: organization } = useSelectedOrganizationQuery()
|
|
const { mutate: sendEvent } = useSendEventMutation()
|
|
const aiSnap = useAiAssistantStateSnapshot()
|
|
const { openSidebar } = useSidebarManagerSnapshot()
|
|
|
|
const [selectedFramework, setSelectedFramework] = useState<string>(DEFAULT_FRAMEWORK_KEY)
|
|
const workflow: 'no-code' | 'code' | null = value === 'code' || value === 'no-code' ? value : null
|
|
const [previousWorkflow, setPreviousWorkflow] = useState<'no-code' | 'code' | null>(null)
|
|
|
|
const statuses = useGettingStartedProgress()
|
|
|
|
const openAiChat = useCallback(
|
|
(name: string, initialInput: string) => {
|
|
openSidebar(SIDEBAR_KEYS.AI_ASSISTANT)
|
|
aiSnap.newChat({ name, initialInput })
|
|
},
|
|
[aiSnap, openSidebar]
|
|
)
|
|
|
|
const connectPresetLinks = useMemo(() => {
|
|
const basePath = router.asPath.split('?')[0]
|
|
const parent = FRAMEWORKS.find((f) => f.key === selectedFramework)
|
|
if (!parent) {
|
|
return [
|
|
{
|
|
label: 'Connect',
|
|
href: `${basePath}?showConnect=true&connectTab=frameworks`,
|
|
},
|
|
]
|
|
}
|
|
|
|
let using: string | undefined
|
|
let withKey: string | undefined
|
|
if (parent.children && parent.children.length > 0) {
|
|
// prefer App Router for Nextjs by default
|
|
if (parent.key === 'nextjs') {
|
|
const appChild = parent.children.find((c) => c.key === 'app') || parent.children[0]
|
|
using = appChild?.key
|
|
withKey = appChild?.children?.[0]?.key
|
|
} else {
|
|
using = parent.children[0]?.key
|
|
withKey = parent.children[0]?.children?.[0]?.key
|
|
}
|
|
}
|
|
|
|
const qs = new URLSearchParams({
|
|
showConnect: 'true',
|
|
connectTab: 'frameworks',
|
|
framework: parent.key,
|
|
})
|
|
if (using) qs.set('using', using)
|
|
if (withKey) qs.set('with', withKey)
|
|
|
|
return [
|
|
{
|
|
label: 'Connect',
|
|
href: `${basePath}?${qs.toString()}`,
|
|
},
|
|
]
|
|
}, [router.asPath, selectedFramework])
|
|
|
|
const connectActions: GettingStartedAction[] = useMemo(() => {
|
|
const selector: GettingStartedAction = {
|
|
label: 'Framework selector',
|
|
component: (
|
|
<FrameworkSelector
|
|
value={selectedFramework}
|
|
onChange={setSelectedFramework}
|
|
items={FRAMEWORKS}
|
|
/>
|
|
),
|
|
}
|
|
|
|
const linkActions: GettingStartedAction[] = connectPresetLinks.map((lnk) => ({
|
|
label: 'Connect',
|
|
href: lnk.href,
|
|
variant: 'primary',
|
|
}))
|
|
|
|
return [selector, ...linkActions]
|
|
}, [connectPresetLinks, selectedFramework])
|
|
|
|
const codeSteps: GettingStartedStep[] = useMemo(
|
|
() =>
|
|
getCodeWorkflowSteps({
|
|
ref,
|
|
openAiChat,
|
|
connectActions,
|
|
statuses,
|
|
}),
|
|
[connectActions, openAiChat, ref, statuses]
|
|
)
|
|
|
|
const noCodeSteps: GettingStartedStep[] = useMemo(
|
|
() =>
|
|
getNoCodeWorkflowSteps({
|
|
ref,
|
|
openAiChat,
|
|
connectActions,
|
|
statuses,
|
|
}),
|
|
[connectActions, openAiChat, ref, statuses]
|
|
)
|
|
|
|
const steps = workflow === 'code' ? codeSteps : workflow === 'no-code' ? noCodeSteps : []
|
|
|
|
return (
|
|
<section className="w-full">
|
|
<div className="flex justify-between items-center mb-6">
|
|
<h3 className="heading-section">Getting started</h3>
|
|
<div className="flex items-center gap-2">
|
|
<ToggleGroup
|
|
type="single"
|
|
value={workflow ?? undefined}
|
|
onValueChange={(v) => {
|
|
if (v) {
|
|
const newWorkflow = v as 'no-code' | 'code'
|
|
setPreviousWorkflow(workflow)
|
|
onChange(newWorkflow)
|
|
sendEvent({
|
|
action: 'home_getting_started_workflow_clicked',
|
|
properties: {
|
|
workflow: newWorkflow === 'no-code' ? 'no_code' : 'code',
|
|
is_switch: previousWorkflow !== null,
|
|
},
|
|
groups: {
|
|
project: project?.ref || '',
|
|
organization: organization?.slug || '',
|
|
},
|
|
})
|
|
}
|
|
}}
|
|
>
|
|
<ToggleGroupItem
|
|
value="no-code"
|
|
aria-label="No-code workflow"
|
|
size="sm"
|
|
className="text-xs gap-2 h-auto"
|
|
>
|
|
<Table2 size={16} strokeWidth={1.5} />
|
|
No-code
|
|
</ToggleGroupItem>
|
|
<ToggleGroupItem
|
|
value="code"
|
|
size="sm"
|
|
aria-label="Code workflow"
|
|
className="text-xs gap-2 h-auto"
|
|
>
|
|
<Code size={16} strokeWidth={1.5} />
|
|
Code
|
|
</ToggleGroupItem>
|
|
</ToggleGroup>
|
|
<Button
|
|
size="tiny"
|
|
type="outline"
|
|
onClick={() => {
|
|
onChange('hidden')
|
|
if (workflow) {
|
|
const completedSteps = (workflow === 'code' ? codeSteps : noCodeSteps).filter(
|
|
(step) => step.status === 'complete'
|
|
).length
|
|
const totalSteps = (workflow === 'code' ? codeSteps : noCodeSteps).length
|
|
sendEvent({
|
|
action: 'home_getting_started_closed',
|
|
properties: {
|
|
workflow: workflow === 'no-code' ? 'no_code' : 'code',
|
|
steps_completed: completedSteps,
|
|
total_steps: totalSteps,
|
|
},
|
|
groups: {
|
|
project: project?.ref || '',
|
|
organization: organization?.slug || '',
|
|
},
|
|
})
|
|
}
|
|
}}
|
|
>
|
|
Dismiss
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
{steps.length === 0 ? (
|
|
<Card className="bg-background/25 border-dashed relative">
|
|
<div className="absolute -inset-16 z-0 opacity-50">
|
|
<img
|
|
src={`${BASE_PATH}/img/reports/bg-grafana-dark.svg`}
|
|
alt="Supabase Grafana"
|
|
className="w-full h-full object-cover object-right hidden dark:block"
|
|
/>
|
|
<img
|
|
src={`${BASE_PATH}/img/reports/bg-grafana-light.svg`}
|
|
alt="Supabase Grafana"
|
|
className="w-full h-full object-cover object-right dark:hidden"
|
|
/>
|
|
<div className="absolute inset-0 bg-gradient-to-r from-background-alternative to-transparent" />
|
|
</div>
|
|
<CardContent className="relative z-10 p-8 md:p-12 grid lg:grid-cols-2 gap-16 items-center">
|
|
<div>
|
|
<h2 className="heading-subSection mb-0 heading-meta text-foreground-light mb-4">
|
|
Choose a preferred workflow
|
|
</h2>
|
|
<p className="text-foreground">
|
|
With Supabase, you have the flexibility to adopt a workflow that works for you. You
|
|
can do everything via the dashboard, or manage your entire project within your own
|
|
codebase.
|
|
</p>
|
|
</div>
|
|
<div className="flex items-stretch gap-4">
|
|
<Button
|
|
size="medium"
|
|
type="outline"
|
|
onClick={() => {
|
|
setPreviousWorkflow(workflow)
|
|
onChange('no-code')
|
|
sendEvent({
|
|
action: 'home_getting_started_workflow_clicked',
|
|
properties: {
|
|
workflow: 'no_code',
|
|
is_switch: previousWorkflow !== null,
|
|
},
|
|
groups: {
|
|
project: project?.ref || '',
|
|
organization: organization?.slug || '',
|
|
},
|
|
})
|
|
}}
|
|
className="block gap-2 h-auto p-4 md:p-8 max-w-80 text-left justify-start bg-background "
|
|
>
|
|
<Table2 size={20} strokeWidth={1.5} className="text-brand" />
|
|
<div className="mt-4">
|
|
<div>No-code</div>
|
|
<div className="text-foreground-light w-full whitespace-normal">
|
|
Ideal for prototyping or getting your project up and running
|
|
</div>
|
|
</div>
|
|
</Button>
|
|
<Button
|
|
size="medium"
|
|
type="outline"
|
|
onClick={() => {
|
|
setPreviousWorkflow(workflow)
|
|
onChange('code')
|
|
sendEvent({
|
|
action: 'home_getting_started_workflow_clicked',
|
|
properties: {
|
|
workflow: 'code',
|
|
is_switch: previousWorkflow !== null,
|
|
},
|
|
groups: {
|
|
project: project?.ref || '',
|
|
organization: organization?.slug || '',
|
|
},
|
|
})
|
|
}}
|
|
className="bg-background block gap-2 h-auto p-4 md:p-8 max-w-80 text-left justify-start"
|
|
>
|
|
<Code size={20} strokeWidth={1.5} className="text-brand" />
|
|
<div className="mt-4">
|
|
<div>Code</div>
|
|
<div className="text-foreground-light whitespace-normal">
|
|
Ideal for teams that want full control of their project
|
|
</div>
|
|
</div>
|
|
</Button>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
) : (
|
|
<GettingStarted
|
|
steps={steps}
|
|
onStepClick={({ stepIndex, stepTitle, actionType, wasCompleted }) => {
|
|
if (workflow) {
|
|
sendEvent({
|
|
action: 'home_getting_started_step_clicked',
|
|
properties: {
|
|
workflow: workflow === 'no-code' ? 'no_code' : 'code',
|
|
step_number: stepIndex + 1,
|
|
step_title: stepTitle,
|
|
action_type: actionType,
|
|
was_completed: wasCompleted,
|
|
},
|
|
groups: {
|
|
project: project?.ref || '',
|
|
organization: organization?.slug || '',
|
|
},
|
|
})
|
|
}
|
|
}}
|
|
/>
|
|
)}
|
|
</section>
|
|
)
|
|
}
|