Files
supabase/apps/studio/components/interfaces/Database/Indexes/Indexes.tsx
Gildas Garcia 96d43099bb chore: refactor Button API so that it can be used a standard button (#46880)
## Problem

Our `<Button>` component breaks the default `button` contract by
redefining the `type` prop to set its variant (`primary`, `default`,
etc) instead of the button type (`submit`, `button`, etc).
This is confusing and forces to write more code when using it with
shadcn components that expect/inject the standard button props.

## Solution

- rename the `type` prop to `variant`
- rename the `htmlType` prop to `type`
- propagate the changes where necessary
- format code

## How to test

As this is just prop renaming, if it builds it's ok

---------

Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
2026-06-16 23:59:58 +02:00

354 lines
13 KiB
TypeScript

import { useParams } from 'common'
import { sortBy } from 'lodash'
import { AlertCircle, Search, Trash } from 'lucide-react'
import { parseAsBoolean, parseAsString, useQueryState } from 'nuqs'
import { useEffect, useRef, useState } from 'react'
import { toast } from 'sonner'
import {
Button,
Card,
SidePanel,
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from 'ui'
import { Input } from 'ui-patterns/DataInputs/Input'
import { ConfirmationModal } from 'ui-patterns/Dialogs/ConfirmationModal'
import { GenericSkeletonLoader, ShimmeringLoader } from 'ui-patterns/ShimmeringLoader'
import { ProtectedSchemaWarning } from '../ProtectedSchemaWarning'
import { CreateIndexSidePanel } from './CreateIndexSidePanel'
import AlertError from '@/components/ui/AlertError'
import CodeEditor from '@/components/ui/CodeEditor/CodeEditor'
import SchemaSelector from '@/components/ui/SchemaSelector'
import { Shortcut } from '@/components/ui/Shortcut'
import { useDatabaseIndexDeleteMutation } from '@/data/database-indexes/index-delete-mutation'
import { useIndexesQuery, type DatabaseIndex } from '@/data/database-indexes/indexes-query'
import { useSchemasQuery } from '@/data/database/schemas-query'
import { useQuerySchemaState } from '@/hooks/misc/useSchemaQueryState'
import { useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
import { useIsProtectedSchema } from '@/hooks/useProtectedSchemas'
import { onSearchInputEscape } from '@/lib/keyboard'
import { SHORTCUT_IDS } from '@/state/shortcuts/registry'
import { useShortcut } from '@/state/shortcuts/useShortcut'
export const Indexes = () => {
const { data: project } = useSelectedProjectQuery()
const { schema: urlSchema, table } = useParams()
const [search, setSearch] = useQueryState('search', parseAsString.withDefault(''))
const [schemaSelectorOpen, setSchemaSelectorOpen] = useState(false)
const searchInputRef = useRef<HTMLInputElement>(null)
const { selectedSchema, setSelectedSchema } = useQuerySchemaState()
const {
data: allIndexes,
error: indexesError,
isPending: isLoadingIndexes,
isSuccess: isSuccessIndexes,
isError: isErrorIndexes,
} = useIndexesQuery({
schema: selectedSchema,
projectRef: project?.ref,
connectionString: project?.connectionString,
})
const [showCreateIndex, setShowCreateIndex] = useQueryState(
'new',
parseAsBoolean.withDefault(false)
)
const [editIndexId, setEditIndexId] = useQueryState('edit', parseAsString)
const selectedIndex = allIndexes?.find((idx) => idx.name === editIndexId)
const [deleteIndexId, setDeleteIndexId] = useQueryState('delete', parseAsString)
const selectedIndexToDelete = allIndexes?.find((idx) => idx.name === deleteIndexId)
const {
data: schemas,
isPending: isLoadingSchemas,
isSuccess: isSuccessSchemas,
isError: isErrorSchemas,
} = useSchemasQuery({
projectRef: project?.ref,
connectionString: project?.connectionString,
})
const {
mutate: deleteIndex,
isPending: isExecuting,
isSuccess: isSuccessDelete,
} = useDatabaseIndexDeleteMutation({
onSuccess: async () => {
setDeleteIndexId(null)
toast.success('Successfully deleted index')
},
})
const { isSchemaLocked } = useIsProtectedSchema({ schema: selectedSchema })
useShortcut(
SHORTCUT_IDS.LIST_PAGE_FOCUS_SEARCH,
() => {
searchInputRef.current?.focus()
searchInputRef.current?.select()
},
{ label: 'Search indexes' }
)
useShortcut(SHORTCUT_IDS.LIST_PAGE_RESET_FILTERS, () => {
setSearch('')
})
const sortedIndexes = sortBy(allIndexes ?? [], (index) => index.name.toLocaleLowerCase())
const indexes =
search.length > 0
? sortedIndexes.filter((index) => index.name.includes(search) || index.table.includes(search))
: sortedIndexes
const onConfirmDeleteIndex = (index: DatabaseIndex) => {
if (!project) return console.error('Project is required')
deleteIndex({
projectRef: project.ref,
connectionString: project.connectionString,
name: index.name,
schema: selectedSchema,
})
}
useEffect(() => {
if (urlSchema !== undefined) {
const schema = schemas?.find((s) => s.name === urlSchema)
if (schema !== undefined) setSelectedSchema(schema.name)
}
}, [urlSchema, isSuccessSchemas])
useEffect(() => {
if (table !== undefined) setSearch(table)
}, [table])
useEffect(() => {
if (isSuccessIndexes && !!editIndexId && !selectedIndex) {
toast('Index not found')
setEditIndexId(null)
}
}, [isSuccessIndexes, editIndexId, selectedIndex, setEditIndexId])
useEffect(() => {
if (isSuccessIndexes && !!deleteIndexId && !selectedIndexToDelete && !isSuccessDelete) {
toast('Index not found')
setDeleteIndexId(null)
}
}, [isSuccessIndexes, deleteIndexId, selectedIndexToDelete, isSuccessDelete, setDeleteIndexId])
return (
<>
<div className="pb-8">
<div className="flex flex-col gap-y-4">
<div className="flex items-center gap-2 flex-wrap">
{isLoadingSchemas && <ShimmeringLoader className="w-[260px]" />}
{isErrorSchemas && (
<div className="w-[260px] text-foreground-light text-sm border px-3 py-1.5 rounded-sm flex items-center space-x-2">
<AlertCircle strokeWidth={2} size={16} />
<p>Failed to load schemas</p>
</div>
)}
{isSuccessSchemas && (
<Shortcut
id={SHORTCUT_IDS.LIST_PAGE_FOCUS_SCHEMA}
onTrigger={() => setSchemaSelectorOpen(true)}
side="bottom"
tooltipOpen={schemaSelectorOpen ? false : undefined}
>
<SchemaSelector
className="w-full lg:w-[180px]"
size="tiny"
showError={false}
selectedSchemaName={selectedSchema}
onSelectSchema={setSelectedSchema}
open={schemaSelectorOpen}
onOpenChange={setSchemaSelectorOpen}
/>
</Shortcut>
)}
<Input
ref={searchInputRef}
size="tiny"
value={search}
className="w-full lg:w-52"
onChange={(e) => setSearch(e.target.value)}
onKeyDown={onSearchInputEscape(search, setSearch)}
placeholder="Search for an index"
icon={<Search />}
/>
{!isSchemaLocked && (
<Shortcut
id={SHORTCUT_IDS.LIST_PAGE_NEW_ITEM}
label="Create new index"
onTrigger={() => setShowCreateIndex(true)}
options={{ enabled: isSuccessSchemas }}
side="bottom"
>
<Button
className="ml-auto grow lg:grow-0"
variant="primary"
onClick={() => setShowCreateIndex(true)}
disabled={!isSuccessSchemas}
>
Create index
</Button>
</Shortcut>
)}
</div>
{isSchemaLocked && <ProtectedSchemaWarning schema={selectedSchema} entity="indexes" />}
{isLoadingIndexes && <GenericSkeletonLoader />}
{isErrorIndexes && (
<AlertError error={indexesError as any} subject="Failed to retrieve database indexes" />
)}
{isSuccessIndexes && (
<div className="w-full overflow-hidden">
<Card>
<Table>
<TableHeader>
<TableRow>
<TableHead key="table">Table</TableHead>
<TableHead key="columns">Columns</TableHead>
<TableHead key="name">Name</TableHead>
<TableHead key="buttons" />
</TableRow>
</TableHeader>
<TableBody>
{indexes.length === 0 && search.length === 0 && (
<TableRow>
<TableCell colSpan={4}>
<p className="text-sm text-foreground">No indexes created yet</p>
<p className="text-sm text-foreground-light">
There are no indexes found in the schema "{selectedSchema}"
</p>
</TableCell>
</TableRow>
)}
{indexes.length === 0 && search.length > 0 && (
<TableRow>
<TableCell colSpan={4}>
<p className="text-sm text-foreground">No results found</p>
<p className="text-sm text-foreground-light">
Your search for "{search}" did not return any results
</p>
</TableCell>
</TableRow>
)}
{indexes.length > 0 &&
indexes.map((index) => (
<TableRow key={index.name}>
<TableCell>
<p title={index.table}>{index.table}</p>
</TableCell>
<TableCell>
<p title={index.columns}>{index.columns}</p>
</TableCell>
<TableCell>
<p title={index.name}>{index.name}</p>
</TableCell>
<TableCell>
<div className="flex justify-end items-center space-x-2">
<Button variant="default" onClick={() => setEditIndexId(index.name)}>
View definition
</Button>
{!isSchemaLocked && (
<Button
aria-label="Delete index"
variant="text"
className="px-1"
icon={<Trash />}
onClick={() => setDeleteIndexId(index.name)}
/>
)}
</div>
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</Card>
</div>
)}
</div>
</div>
<SidePanel
size="xlarge"
visible={!!selectedIndex}
header={
<>
<span>Index:</span>
<code className="text-sm ml-2">{selectedIndex?.name}</code>
</>
}
onCancel={() => setEditIndexId(null)}
>
<div className="h-full">
<div className="relative h-full">
<CodeEditor
isReadOnly
id={selectedIndex?.name ?? ''}
language="pgsql"
defaultValue={selectedIndex?.definition ?? ''}
/>
</div>
</div>
</SidePanel>
<CreateIndexSidePanel visible={showCreateIndex} onClose={() => setShowCreateIndex(false)} />
<ConfirmationModal
variant="warning"
size="medium"
loading={isExecuting}
visible={!!selectedIndexToDelete}
title={
<>
Confirm to delete index{' '}
<code className="text-code-inline">{selectedIndexToDelete?.name}</code>
</>
}
confirmLabel="Confirm delete"
confirmLabelLoading="Deleting..."
onConfirm={() =>
selectedIndexToDelete !== undefined ? onConfirmDeleteIndex(selectedIndexToDelete) : {}
}
onCancel={() => setDeleteIndexId(null)}
alert={{
title: 'This action cannot be undone',
description:
'Deleting an index that is still in use will cause queries to slow down, and in some cases causing significant performance issues.',
}}
className="pt-0"
>
<ul className="mt-4 space-y-5">
<li className="flex gap-3">
<div>
<strong className="text-sm">Before deleting this index, consider:</strong>
<ul className="space-y-2 mt-2 text-sm text-foreground-light">
<li className="list-disc ml-6">This index is no longer in use</li>
<li className="list-disc ml-6">
The table which the index is on is not currently in use, as dropping an index
requires a short exclusive access lock on the table.
</li>
</ul>
</div>
</li>
</ul>
</ConfirmationModal>
</>
)
}