Files
supabase/apps/studio/components/interfaces/Database/Tables/ColumnList.tsx
Joshen Lim cab0585533 Fe 1799/consolidate to useselectedprojectquery and (#37684)
* Replace all usage of useProjectContext with useSelectedProjectQuery

* Replace all usage of useSelectedProject with useSelectedProjectQuery

* Replace all usage of useProjectByRef with useProjectByRefQuery

* Replace all usage of useSelectedOrganization with useSelectedOrganizationQuery

* Deprecate useSelectedProject, useSelectedOrganization, and useProjectByRef hooks

* Deprecate ProjecContext
2025-08-06 10:53:10 +07:00

218 lines
8.0 KiB
TypeScript

import { PermissionAction } from '@supabase/shared-types/out/constants'
import { noop } from 'lodash'
import { Check, ChevronLeft, Edit, MoreVertical, Plus, Search, Trash, X } from 'lucide-react'
import Link from 'next/link'
import { useState } from 'react'
import { useParams } from 'common'
import NoSearchResults from 'components/to-be-cleaned/NoSearchResults'
import Table from 'components/to-be-cleaned/Table'
import AlertError from 'components/ui/AlertError'
import { ButtonTooltip } from 'components/ui/ButtonTooltip'
import { GenericSkeletonLoader } from 'components/ui/ShimmeringLoader'
import { useTableEditorQuery } from 'data/table-editor/table-editor-query'
import { isTableLike } from 'data/table-editor/table-editor-types'
import { useCheckPermissions } from 'hooks/misc/useCheckPermissions'
import { useSelectedProjectQuery } from 'hooks/misc/useSelectedProject'
import { useIsProtectedSchema } from 'hooks/useProtectedSchemas'
import {
Button,
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuTrigger,
Input,
Tooltip,
TooltipContent,
TooltipTrigger,
} from 'ui'
import { ProtectedSchemaWarning } from '../ProtectedSchemaWarning'
interface ColumnListProps {
onAddColumn: () => void
onEditColumn: (column: any) => void
onDeleteColumn: (column: any) => void
}
const ColumnList = ({
onAddColumn = noop,
onEditColumn = noop,
onDeleteColumn = noop,
}: ColumnListProps) => {
const { id: _id, ref } = useParams()
const id = _id ? Number(_id) : undefined
const { data: project } = useSelectedProjectQuery()
const {
data: selectedTable,
error,
isError,
isLoading,
isSuccess,
} = useTableEditorQuery({
projectRef: project?.ref,
connectionString: project?.connectionString,
id,
})
const [filterString, setFilterString] = useState<string>('')
const isTableEntity = isTableLike(selectedTable)
const columns =
(filterString.length === 0
? selectedTable?.columns ?? []
: selectedTable?.columns?.filter((column) => column.name.includes(filterString))) ?? []
const { isSchemaLocked } = useIsProtectedSchema({ schema: selectedTable?.schema ?? '' })
const canUpdateColumns = useCheckPermissions(PermissionAction.TENANT_SQL_ADMIN_WRITE, 'columns')
return (
<div className="space-y-4">
<div className="flex items-center justify-between">
<div className="flex items-center space-x-4">
<Button asChild type="outline" icon={<ChevronLeft />} style={{ padding: '5px' }}>
<Link href={`/project/${ref}/database/tables`} />
</Button>
<Input
size="small"
placeholder="Filter columns"
value={filterString}
onChange={(e: any) => setFilterString(e.target.value)}
icon={<Search size={12} />}
/>
</div>
{!isSchemaLocked && isTableEntity && (
<ButtonTooltip
icon={<Plus />}
disabled={!canUpdateColumns}
onClick={() => onAddColumn()}
tooltip={{
content: {
side: 'bottom',
text: !canUpdateColumns
? 'You need additional permissions to create columns'
: undefined,
},
}}
>
New column
</ButtonTooltip>
)}
</div>
{isSchemaLocked && (
<ProtectedSchemaWarning schema={selectedTable?.schema ?? ''} entity="columns" />
)}
{isLoading && <GenericSkeletonLoader />}
{isError && (
<AlertError
error={error as any}
subject={`Failed to retrieve columns for table "${selectedTable?.schema}.${selectedTable?.name}"`}
/>
)}
{isSuccess && (
<>
{columns.length === 0 ? (
<NoSearchResults />
) : (
<div>
<Table
head={[
<Table.th key="name">Name</Table.th>,
<Table.th key="description" className="hidden lg:table-cell">
Description
</Table.th>,
<Table.th key="type">Data Type</Table.th>,
<Table.th key="format">Format</Table.th>,
<Table.th key="format" className="text-center">
Nullable
</Table.th>,
<Table.th key="buttons"></Table.th>,
]}
body={columns.map((x) => (
<Table.tr className="border-t" key={x.name}>
<Table.td>
<p>{x.name}</p>
</Table.td>
<Table.td className="break-all whitespace-normal hidden xl:table-cell">
{x.comment !== null ? (
<p title={x.comment}>{x.comment}</p>
) : (
<p className="text-border-stronger">No description</p>
)}
</Table.td>
<Table.td>
<code className="text-xs">{x.data_type}</code>
</Table.td>
<Table.td className="font-mono text-xs">
<code className="text-xs">{x.format}</code>
</Table.td>
<Table.td className="font-mono text-xs">
{x.is_nullable ? (
<Check size={16} className="mx-auto" />
) : (
<X size={16} className="mx-auto" />
)}
</Table.td>
<Table.td className="text-right">
{!isSchemaLocked && isTableEntity && (
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button type="default" className="px-1" icon={<MoreVertical />} />
</DropdownMenuTrigger>
<DropdownMenuContent side="bottom" align="end" className="w-32">
<Tooltip>
<TooltipTrigger>
<DropdownMenuItem
disabled={!canUpdateColumns}
onClick={() => onEditColumn(x)}
className="space-x-2"
>
<Edit size={12} />
<p>Edit column</p>
</DropdownMenuItem>
</TooltipTrigger>
{!canUpdateColumns && (
<TooltipContent side="bottom">
Additional permissions required to edit column
</TooltipContent>
)}
</Tooltip>
<Tooltip>
<TooltipTrigger>
<DropdownMenuItem
disabled={!canUpdateColumns || isSchemaLocked}
onClick={() => onDeleteColumn(x)}
className="space-x-2"
>
<Trash stroke="red" size={12} />
<p>Delete column</p>
</DropdownMenuItem>
</TooltipTrigger>
{!canUpdateColumns && (
<TooltipContent side="bottom">
Additional permissions required to delete column
</TooltipContent>
)}
</Tooltip>
</DropdownMenuContent>
</DropdownMenu>
)}
</Table.td>
</Table.tr>
))}
/>
</div>
)}
</>
)}
</div>
)
}
export default ColumnList