ruvector/studio/components/layouts/IntegrationsLayout/IntegrationWindowLayout.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

97 lines
3 KiB
TypeScript

import Link from 'next/link'
import { PropsWithChildren, ReactNode, forwardRef } from 'react'
import { LoadingLine, cn } from 'ui'
import { withAuth } from 'hooks/misc/withAuth'
import { BASE_PATH } from 'lib/constants'
import { Book, LifeBuoy, X } from 'lucide-react'
import { ScaffoldContainer } from '../Scaffold'
export type IntegrationWindowLayoutProps = {
title: string
integrationIcon: ReactNode
loading?: boolean
docsHref?: string
}
const IntegrationWindowLayout = ({
title,
integrationIcon,
children,
loading = false,
docsHref,
}: PropsWithChildren<IntegrationWindowLayoutProps>) => {
return (
<div className="flex w-full flex-col">
<Header title={title} integrationIcon={integrationIcon} />
<LoadingLine loading={loading} />
<main className="overflow-auto flex flex-col h-full bg">{children}</main>
<ScaffoldContainer className="bg-studio flex flex-row gap-6 py-6 border-t">
{docsHref && (
<Link
href={docsHref}
target="_blank"
rel="noopener noreferrer"
className="flex items-center gap-2 text-xs text-foreground-light hover:text"
>
<Book size={16} />
Docs
</Link>
)}
<Link
href={'https://supabase.com/support'}
target="_blank"
rel="noopener noreferrer"
className="flex items-center gap-2 text-xs text-light hover:text"
>
<LifeBuoy size={16} />
Support
</Link>
</ScaffoldContainer>
</div>
)
}
const INTEGRATION_LAYOUT_MAX_WIDTH = '' // 'max-w-[720px]'
export default withAuth(IntegrationWindowLayout)
export const IntegrationWindowLayoutWithoutAuth = IntegrationWindowLayout
export type HeaderProps = {
title: string
integrationIcon: ReactNode
}
const Header = ({ title, integrationIcon }: HeaderProps) => {
return (
<div className="bg">
<ScaffoldContainer className={cn('py-3', INTEGRATION_LAYOUT_MAX_WIDTH)}>
<div className="flex items-center gap-6 w-full">
<div className="flex gap-2 items-center">
<div className="bg-white shadow border rounded p-1 w-8 h-8 flex justify-center items-center">
<img src={`${BASE_PATH}/img/supabase-logo.svg`} alt="Supabase" className="w-4" />
</div>
<X className="text-border-stronger" strokeWidth={2} size={16} />
{integrationIcon}
</div>
<span className="text-sm" title={title}>
{title}
</span>
</div>
</ScaffoldContainer>
</div>
)
}
const maxWidthClasses = 'mx-auto w-full max-w-[1600px]'
const paddingClasses = 'px-6 lg:px-14 xl:px-28 2xl:px-32'
export const IntegrationScaffoldContainer = forwardRef<
HTMLDivElement,
React.HTMLAttributes<HTMLDivElement>
>(({ className, ...props }, ref) => {
return <div ref={ref} {...props} className={cn(maxWidthClasses, paddingClasses, className)} />
})
IntegrationScaffoldContainer.displayName = 'IntegrationScaffoldContainer'