mirror of
https://github.com/supabase/supabase.git
synced 2026-05-07 23:19:23 +08:00
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>
122 lines
3.6 KiB
TypeScript
122 lines
3.6 KiB
TypeScript
'use client'
|
|
|
|
import { File } from 'lucide-react'
|
|
import { useState } from 'react'
|
|
import { flattenTree, TreeView, TreeViewItem } from 'ui'
|
|
import { CodeBlock } from 'ui-patterns/CodeBlock'
|
|
|
|
import { RegistryNode } from '@/lib/process-registry'
|
|
|
|
interface BlockItemCodeProps {
|
|
files: RegistryNode[]
|
|
}
|
|
|
|
interface TreeNode {
|
|
name: string
|
|
children: TreeNode[]
|
|
metadata: { path: string }
|
|
}
|
|
|
|
const flattenChildren = (files: RegistryNode[]): TreeNode[] => {
|
|
return files.map(
|
|
(node): TreeNode => ({
|
|
name: node.name,
|
|
children: node.children ? flattenChildren(node.children) : [],
|
|
metadata: { path: node.path },
|
|
})
|
|
)
|
|
}
|
|
|
|
const findFirstFile = (nodes: RegistryNode[]): RegistryNode | null => {
|
|
for (const node of nodes) {
|
|
if (node.type === 'file') {
|
|
return node
|
|
}
|
|
if (node.children) {
|
|
const foundFile = findFirstFile(node.children)
|
|
if (foundFile) {
|
|
return foundFile
|
|
}
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
export function BlockItemCode({ files }: BlockItemCodeProps) {
|
|
// Find the first file to select by default
|
|
const [selectedFile, setSelectedFile] = useState<RegistryNode | null>(findFirstFile(files))
|
|
const flattenedData = flattenTree({ name: '', children: flattenChildren(files) })
|
|
|
|
// Handle file selection from the TreeView
|
|
const handleNodeSelect = (element: any) => {
|
|
const findFileByPath = (nodes: RegistryNode[], path: string): RegistryNode | null => {
|
|
for (const node of nodes) {
|
|
if (node.path === path) {
|
|
return node
|
|
}
|
|
if (node.children) {
|
|
const found = findFileByPath(node.children, path)
|
|
if (found) {
|
|
return found
|
|
}
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
const filePath = element.metadata.path
|
|
const foundFile = findFileByPath(files, filePath)
|
|
|
|
if (foundFile?.type === 'directory') return
|
|
|
|
setSelectedFile(foundFile || null)
|
|
}
|
|
|
|
return (
|
|
<div className="flex mt-4 border rounded-lg overflow-hidden h-[652px] not-prose">
|
|
{/* File browser sidebar */}
|
|
<div className="w-64 grow-0 shrink-0 flex-0 py-2 border-r bg-muted/30 overflow-y-auto">
|
|
<TreeView
|
|
data={flattenedData}
|
|
aria-label="file browser"
|
|
className="w-full"
|
|
defaultExpandedIds={flattenedData.filter((n) => n.children?.length).map((n) => n.id)}
|
|
defaultSelectedIds={flattenedData
|
|
.filter((n) => n.metadata?.path === selectedFile?.path)
|
|
.map((n) => n.id)}
|
|
onNodeSelect={({ element }) => handleNodeSelect(element)}
|
|
nodeRenderer={({ element, isBranch, isExpanded, getNodeProps, level, isSelected }) => (
|
|
<TreeViewItem
|
|
{...getNodeProps()}
|
|
isExpanded={isExpanded}
|
|
isBranch={isBranch}
|
|
isSelected={isSelected}
|
|
level={level}
|
|
icon={<File strokeWidth={1.5} size={16} className="shrink-0" />}
|
|
name={element.name}
|
|
className="gap-1.5"
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
|
|
{/* Code display area */}
|
|
{selectedFile?.content ? (
|
|
<CodeBlock
|
|
wrapperClassName="w-full"
|
|
className="h-full max-w-none w-full! flex-1 font-mono text-xs rounded-none border-none"
|
|
language="ts"
|
|
>
|
|
{selectedFile?.content}
|
|
</CodeBlock>
|
|
) : (
|
|
<div className="flex items-center justify-center h-full text-muted-foreground">
|
|
<div className="flex flex-col items-center gap-2">
|
|
<p>No file selected or file content unavailable</p>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
)
|
|
}
|