Files
supabase/apps/www/components/Realtime/example-layout.tsx
Ivan Vasilov 56de26fe22 chore: Migrate the monorepo to use Tailwind v4 (#45318)
This PR migrates the whole monorepo to use Tailwind v4:
- Removed `@tailwindcss/container-queries` plugin since it's included by
default in v4,
- Bump all instances of Tailwind to v4. Made minimal changes to the
shared config to remove non-supported features (`alpha` mentions),
- Migrate all apps to be compatible with v4 configs,
- Fix the `typography.css` import in 3 apps,
- Add missing rules which were included by default in v3,
- Run `pnpm dlx @tailwindcss/upgrade` on all apps, which renames a lot
of classes
- Rename all misnamed classes according to
https://tailwindcss.com/docs/upgrade-guide#renamed-utilities in all
apps.

---------

Co-authored-by: Jordi Enric <jordi.err@gmail.com>
2026-04-30 10:53:24 +00:00

116 lines
3.6 KiB
TypeScript

'use client'
import { ChevronLeft, ChevronRight, Code, Eye } from 'lucide-react'
import { useState } from 'react'
import { Button } from 'ui'
import { CodeBlock } from 'ui-patterns/CodeBlock'
import SandpackWrapper from './sandpack'
export type ExampleLayoutProps = {
appJsCode: string
files: Record<string, string>
dependencies?: Record<string, string>
title?: string
description?: string
onPrevious?: () => void
onNext?: () => void
}
export default function ExampleLayout({
appJsCode,
files,
dependencies = {},
title = 'App.js',
description = 'This is an example of a realtime app.',
onPrevious,
onNext,
}: ExampleLayoutProps) {
const [viewMode, setViewMode] = useState<'code' | 'preview'>('preview')
const formattedCode = `// # ${title}
// ${description}
${appJsCode}`
return (
<div className="h-full w-full flex flex-col overflow-hidden">
{/* Header Bar */}
<div className="grid grid-cols-3 items-center px-2 py-2 border-b border-muted">
{/* Left side: Code/Preview Toggle */}
<div>
<div className="flex w-fit gap-1">
<Button
type="text"
onClick={() => setViewMode('code')}
icon={<Code size={14} />}
size="tiny"
className={`${viewMode === 'code' ? 'bg-surface-200 text-foreground' : ''}`}
aria-label="Show code"
>
<span className="hidden sm:inline">Code</span>
</Button>
<Button
type="text"
onClick={() => setViewMode('preview')}
size="tiny"
icon={<Eye size={14} />}
className={`${viewMode === 'preview' ? 'bg-surface-200 text-foreground' : ''}`}
aria-label="Show preview"
>
<span className="hidden sm:inline">Preview</span>
</Button>
</div>
</div>
{/* Center: Title */}
<div className="text-sm font-medium truncate mx-2 text-center">{title}</div>
{/* Right: Navigation */}
<div className="flex gap-1 justify-end">
<Button
type="outline"
onClick={onPrevious}
disabled={!onPrevious}
className="p-1 rounded-sm hover:bg-surface-200 text-foreground-light disabled:opacity-50 disabled:cursor-not-allowed"
aria-label="Previous example"
>
<ChevronLeft size={16} />
</Button>
<Button
type="outline"
onClick={onNext}
disabled={!onNext}
className="p-1 rounded-sm hover:bg-surface-200 text-foreground-light disabled:opacity-50 disabled:cursor-not-allowed"
aria-label="Next example"
>
<ChevronRight size={16} />
</Button>
</div>
</div>
{viewMode === 'preview' ? (
/* Previews - shown when in preview mode */
<div className="h-full grid grid-cols-1 lg:grid-cols-2 divide-y lg:divide-y-0 lg:divide-x divide-muted">
<div className="h-96 lg:h-full">
<SandpackWrapper files={files} dependencies={dependencies} />
</div>
<div className="h-96 lg:h-full">
<SandpackWrapper files={files} dependencies={dependencies} />
</div>
</div>
) : (
/* Code view - shown when in code mode */
<CodeBlock
hideLineNumbers
wrapperClassName="w-full"
className="bg-transparent! h-[500px] lg:h-full overflow-auto p-8 border-none rounded-none"
wrapLines
language="jsx"
>
{formattedCode}
</CodeBlock>
)}
</div>
)
}